123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364 |
- /*
- * File : filerw.c
- * This file is part of RT-Thread RTOS
- * COPYRIGHT (C) 2006 - 2009, RT-Thread Development Team
- *
- * The license and distribution terms for this file may be
- * found in the file LICENSE in this distribution or at
- * http://www.rt-thread.org/license/LICENSE
- *
- * Change Logs:
- * Date Author Notes
- * 2009-10-16 Bernard first version
- */
- #include <rtgui/rtgui_system.h>
- #include <rtgui/filerw.h>
- #ifdef RTGUI_USING_DFS_FILERW
- /* standard file read/write */
- struct rtgui_filerw_stdio
- {
- /* inherit from rtgui_filerw */
- struct rtgui_filerw parent;
- int fd;
- rt_bool_t eof;
- };
- static int stdio_seek(struct rtgui_filerw *context, rt_off_t offset, int whence)
- {
- struct rtgui_filerw_stdio *stdio_filerw = (struct rtgui_filerw_stdio *)context;
- int stdio_whence[3] = {SEEK_SET, SEEK_CUR, SEEK_END};
- if (whence < RTGUI_FILE_SEEK_SET || whence > RTGUI_FILE_SEEK_END)
- {
- return -1;
- }
- return lseek(stdio_filerw->fd, offset, stdio_whence[whence]);
- }
- static int stdio_read(struct rtgui_filerw *context, void *ptr, rt_size_t size, rt_size_t maxnum)
- {
- int result;
- struct rtgui_filerw_stdio *stdio_filerw = (struct rtgui_filerw_stdio *)context;
- /* end of file */
- if (stdio_filerw->eof == RT_TRUE) return -1;
- result = read(stdio_filerw->fd, ptr, size * maxnum);
- if (result == 0) stdio_filerw->eof = RT_TRUE;
- return result;
- }
- static int stdio_write(struct rtgui_filerw *context, const void *ptr, rt_size_t size, rt_size_t num)
- {
- struct rtgui_filerw_stdio *stdio_filerw = (struct rtgui_filerw_stdio *)context;
- return write(stdio_filerw->fd, (char *)ptr, size * num);
- }
- static int stdio_tell(struct rtgui_filerw *context)
- {
- struct rtgui_filerw_stdio *stdio_filerw = (struct rtgui_filerw_stdio *)context;
- return lseek(stdio_filerw->fd, 0, SEEK_CUR);
- }
- static int stdio_eof(struct rtgui_filerw *context)
- {
- int result;
- struct rtgui_filerw_stdio *stdio_filerw = (struct rtgui_filerw_stdio *)context;
- if (stdio_filerw->eof == RT_TRUE) result = 1;
- else result = -1;
- return result;
- }
- static int stdio_close(struct rtgui_filerw *context)
- {
- struct rtgui_filerw_stdio *stdio_filerw = (struct rtgui_filerw_stdio *)context;
- if (stdio_filerw)
- {
- close(stdio_filerw->fd);
- rtgui_free(stdio_filerw);
- return 0;
- }
- return -1;
- }
- #endif
- /* memory file read/write */
- struct rtgui_filerw_mem
- {
- /* inherit from rtgui_filerw */
- struct rtgui_filerw parent;
- const rt_uint8_t *mem_base, *mem_position, *mem_end;
- };
- static int mem_seek(struct rtgui_filerw *context, rt_off_t offset, int whence)
- {
- const rt_uint8_t *newpos;
- struct rtgui_filerw_mem *mem = (struct rtgui_filerw_mem *)context;
- RT_ASSERT(mem != RT_NULL);
- switch (whence)
- {
- case RTGUI_FILE_SEEK_SET:
- newpos = mem->mem_base + offset;
- break;
- case RTGUI_FILE_SEEK_CUR:
- newpos = mem->mem_position + offset;
- break;
- case RTGUI_FILE_SEEK_END:
- newpos = mem->mem_end + offset;
- break;
- default:
- return -1;
- }
- if (newpos < mem->mem_base)
- newpos = mem->mem_base;
- if (newpos > mem->mem_end)
- newpos = mem->mem_end;
- mem->mem_position = newpos;
- return mem->mem_position - mem->mem_base;
- }
- static int mem_read(struct rtgui_filerw *context, void *ptr, rt_size_t size, rt_size_t maxnum)
- {
- int total_bytes;
- int mem_available;
- struct rtgui_filerw_mem *mem = (struct rtgui_filerw_mem *)context;
- total_bytes = (maxnum * size);
- if ((maxnum <= 0) || (size <= 0) || ((total_bytes / maxnum) != size))
- {
- return -1;
- }
- mem_available = mem->mem_end - mem->mem_position;
- if (total_bytes > mem_available)
- total_bytes = mem_available;
- rt_memcpy(ptr, mem->mem_position, total_bytes);
- mem->mem_position += total_bytes;
- return (total_bytes / size);
- }
- static int mem_write(struct rtgui_filerw *context, const void *ptr, rt_size_t size, rt_size_t num)
- {
- return 0; /* not support memory write */
- }
- static int mem_tell(struct rtgui_filerw *context)
- {
- struct rtgui_filerw_mem *mem = (struct rtgui_filerw_mem *)context;
- return mem->mem_position - mem->mem_base;
- }
- static int mem_eof(struct rtgui_filerw *context)
- {
- struct rtgui_filerw_mem *mem = (struct rtgui_filerw_mem *)context;
- return mem->mem_position >= mem->mem_end;
- }
- static int mem_close(struct rtgui_filerw *context)
- {
- struct rtgui_filerw_mem *mem = (struct rtgui_filerw_mem *)context;
- if (mem != RT_NULL)
- {
- rtgui_free(mem);
- return 0;
- }
- return -1;
- }
- const rt_uint8_t *rtgui_filerw_mem_getdata(struct rtgui_filerw *context)
- {
- struct rtgui_filerw_mem *mem = (struct rtgui_filerw_mem *)context;
- /* check whether it's a memory filerw */
- if (mem->parent.read != mem_read) return RT_NULL;
- return mem->mem_base;
- }
- /* file read/write public interface */
- #ifdef RTGUI_USING_DFS_FILERW
- static int parse_mode(const char *mode)
- {
- int f = 0;
- for (;;)
- {
- switch (*mode)
- {
- case 0:
- return f;
- case 'b':
- f |= O_BINARY;
- break;
- case 'r':
- f = O_RDONLY;
- break;
- case 'w':
- f = O_WRONLY | O_CREAT | O_TRUNC;
- break;
- case 'a':
- f = O_WRONLY | O_CREAT | O_APPEND;
- break;
- case '+':
- f = (f & (~O_WRONLY)) | O_RDWR;
- break;
- }
- ++mode;
- }
- }
- struct rtgui_filerw *rtgui_filerw_create_file(const char *filename, const char *mode)
- {
- int fd;
- struct rtgui_filerw_stdio *rw;
- RT_ASSERT(filename != RT_NULL);
- rw = RT_NULL;
- #ifdef _WIN32_NATIVE
- fd = _open(filename, parse_mode(mode), 0);
- #else
- fd = open(filename, parse_mode(mode), 0);
- #endif
- if (fd >= 0)
- {
- rw = (struct rtgui_filerw_stdio *) rtgui_malloc(sizeof(struct rtgui_filerw_stdio));
- if (rw != RT_NULL)
- {
- rw->parent.seek = stdio_seek;
- rw->parent.read = stdio_read;
- rw->parent.write = stdio_write;
- rw->parent.tell = stdio_tell;
- rw->parent.close = stdio_close;
- rw->parent.eof = stdio_eof;
- rw->fd = fd;
- rw->eof = RT_FALSE;
- }
- }
- return &(rw->parent);
- }
- int rtgui_filerw_unlink(const char *filename)
- {
- #ifdef _WIN32_NATIVE
- return _unlink(filename);
- #else
- #ifndef RT_USING_DFS
- /* no unlink function */
- return -1;
- #else
- return unlink(filename);
- #endif
- #endif
- }
- #endif
- struct rtgui_filerw *rtgui_filerw_create_mem(const rt_uint8_t *mem, rt_size_t size)
- {
- struct rtgui_filerw_mem *rw;
- RT_ASSERT(mem != RT_NULL);
- rw = (struct rtgui_filerw_mem *) rtgui_malloc(sizeof(struct rtgui_filerw_mem));
- if (rw != RT_NULL)
- {
- rw->parent.seek = mem_seek;
- rw->parent.read = mem_read;
- rw->parent.write = mem_write;
- rw->parent.tell = mem_tell;
- rw->parent.eof = mem_eof;
- rw->parent.close = mem_close;
- rw->mem_base = mem;
- rw->mem_position = mem;
- rw->mem_end = mem + size;
- }
- return &(rw->parent);
- }
- int rtgui_filerw_seek(struct rtgui_filerw *context, rt_off_t offset, int whence)
- {
- RT_ASSERT(context != RT_NULL);
- return context->seek(context, offset, whence);
- }
- int rtgui_filerw_read(struct rtgui_filerw *context, void *buffer, rt_size_t size, rt_size_t count)
- {
- RT_ASSERT(context != RT_NULL);
- return context->read(context, buffer, size, count);
- }
- int rtgui_filerw_write(struct rtgui_filerw *context, const void *buffer, rt_size_t size, rt_size_t count)
- {
- RT_ASSERT(context != RT_NULL);
- return context->write(context, buffer, size, count);
- }
- int rtgui_filerw_eof(struct rtgui_filerw *context)
- {
- RT_ASSERT(context != RT_NULL);
- return context->eof(context);
- }
- int rtgui_filerw_tell(struct rtgui_filerw *context)
- {
- RT_ASSERT(context != RT_NULL);
- return context->tell(context);
- }
- int rtgui_filerw_close(struct rtgui_filerw *context)
- {
- int result;
- RT_ASSERT(context != RT_NULL);
- /* close context */
- result = context->close(context);
- if (result != 0)
- {
- /* close file failed */
- return -1;
- }
- return 0;
- }
|