Usermode/libc - stdin caching
authorJohn Hodge <[email protected]>
Thu, 21 Nov 2013 04:46:40 +0000 (12:46 +0800)
committerJohn Hodge <[email protected]>
Thu, 21 Nov 2013 04:46:40 +0000 (12:46 +0800)
Usermode/Libraries/libc.so_src/stdio.c
Usermode/Libraries/libc.so_src/stdio_int.h

index 315f698..372c489 100644 (file)
@@ -40,13 +40,13 @@ void _stdio_init(void)
        // Init FileIO Pointers\r
        stdin = &_iob[0];\r
        stdin->FD = 0;\r
-       stdin->Flags = FILE_FLAG_ALLOC|FILE_FLAG_MODE_READ|FILE_FLAG_LINEBUFFERED;\r
+       stdin->Flags = FILE_FLAG_ALLOC|FILE_FLAG_MODE_READ|FILE_FLAG_LINEBUFFERED|FILE_FLAG_OURBUFFER;\r
        stdin->Buffer = malloc(STDIN_BUFSIZ);\r
        stdin->BufferSpace = STDIN_BUFSIZ;\r
 \r
        stdout = &_iob[1];\r
        stdout->FD = 1;\r
-       stdout->Flags = FILE_FLAG_ALLOC|FILE_FLAG_MODE_WRITE|FILE_FLAG_LINEBUFFERED;\r
+       stdout->Flags = FILE_FLAG_ALLOC|FILE_FLAG_MODE_WRITE|FILE_FLAG_LINEBUFFERED|FILE_FLAG_OURBUFFER;\r
        stdout->Buffer = malloc(STDOUT_BUFSIZ);\r
        stdout->BufferSpace = STDOUT_BUFSIZ;\r
        \r
@@ -151,6 +151,16 @@ EXPORT FILE *freopen(const char *file, const char *mode, FILE *fp)
                return NULL;\r
        }\r
        \r
+       // Default to buffered\r
+       // - Disabled until fseek() is fixed\r
+       #if 0\r
+       fp->BufferOfs = 0;\r
+       fp->BufferPos = 0;\r
+       fp->BufferSpace = BUFSIZ;\r
+       fp->Buffer = malloc( fp->BufferSpace );\r
+       fp->Flags |= FILE_FLAG_OURBUFFER;\r
+       #endif\r
+       \r
        if( (fp->Flags & FILE_FLAG_MODE_MASK) == FILE_FLAG_MODE_APPEND ) {\r
                _SysSeek(fp->FD, 0, SEEK_END);  //SEEK_END\r
        }\r
@@ -220,6 +230,10 @@ EXPORT int fclose(FILE *fp)
        fflush(fp);\r
        if( fp->FD >= 0 ) {\r
                _SysClose(fp->FD);\r
+               if( fp->Buffer && (fp->Flags & FILE_FLAG_OURBUFFER) ) {\r
+                       free(fp->Buffer);\r
+               }\r
+               fp->Buffer = NULL;\r
        }\r
        fp->Flags = 0;\r
        fp->FD = FD_NOTOPEN;\r
@@ -262,9 +276,13 @@ EXPORT int setvbuf(FILE *fp, char *buffer, int mode, size_t size)
                }\r
                // Allocate a buffer if one was not provided\r
                if( !buffer ) {\r
+                       fp->Flags |= FILE_FLAG_OURBUFFER;\r
                        buffer = malloc(size);\r
                        assert(buffer);\r
                }\r
+               else {\r
+                       fp->Flags &= ~FILE_FLAG_OURBUFFER;\r
+               }\r
                \r
                // Set buffer pointer and size\r
                fp->Buffer = buffer;\r
@@ -592,6 +610,37 @@ size_t _fread_memstream(void *ptr, size_t size, size_t num, FILE *fp)
 }\r
 #endif\r
 \r
+size_t _fread_buffered(void *ptr, size_t size, FILE *fp)\r
+{\r
+       _SysDebug("%p: %i-%i <= %i", fp,\r
+               (int)fp->Pos, (int)fp->BufferOfs, (int)fp->BufferPos);\r
+       if( fp->BufferPos > 0 ) {\r
+               assert( fp->Pos - fp->BufferOfs <= fp->BufferPos );\r
+       }\r
+       if( fp->BufferPos == 0 || fp->Pos - fp->BufferOfs == fp->BufferPos )\r
+       {\r
+               int rv = _SysRead(fp->FD, fp->Buffer, fp->BufferSpace);\r
+               if( rv <= 0 ) {\r
+                       fp->Flags |= FILE_FLAG_EOF;\r
+                       return 0;\r
+               }\r
+               \r
+               fp->BufferPos = rv;\r
+               fp->BufferOfs = fp->Pos;\r
+               _SysDebug("%p: Buffered %i at %i", fp, rv, fp->Pos);\r
+       }\r
+       \r
+       size_t  inner_ofs = fp->Pos - fp->BufferOfs;\r
+       if(size > fp->BufferPos - inner_ofs)\r
+               size = fp->BufferPos - inner_ofs;\r
+       \r
+       _SysDebug("%p: Read %i from %i+%i", fp, size,\r
+               (int)fp->BufferOfs, inner_ofs);\r
+       memcpy(ptr, fp->Buffer + inner_ofs, size);\r
+       fp->Pos += size;\r
+       return size;\r
+}\r
+\r
 /**\r
  * \fn EXPORT size_t fread(void *ptr, size_t size, size_t num, FILE *fp)\r
  * \brief Read from a stream\r
@@ -610,6 +659,11 @@ EXPORT size_t fread(void *ptr, size_t size, size_t num, FILE *fp)
                return -1;\r
        }\r
 \r
+       // Don't read if EOF is set\r
+       if( fp->Flags & FILE_FLAG_EOF )\r
+               return 0;\r
+\r
+       \r
        if( fp->FD == FD_MEMFILE ) {\r
                return _fread_memfile(ptr, size, num, fp);\r
        }\r
@@ -618,18 +672,49 @@ EXPORT size_t fread(void *ptr, size_t size, size_t num, FILE *fp)
                errno = EBADF;\r
                return 0;\r
        }\r
-       \r
+\r
        // Standard file\r
-       ret = _SysRead(fp->FD, ptr, size*num);\r
-       if( ret == (size_t)-1)\r
-               return -1;\r
-       if( ret == 0 && size*num > 0 ) {\r
-               fp->Flags |= FILE_FLAG_EOF;\r
-               return 0;\r
+       const size_t    bytes = size*num;\r
+\r
+       // TODO: Buffered reads\r
+       if( fp->BufferSpace )\r
+       {\r
+               size_t  ofs = 0;\r
+               size_t  rv;\r
+               // While not done, and buffered read succeeds\r
+               while( ofs < bytes && (rv = _fread_buffered((char*)ptr + ofs, bytes - ofs, fp)) != 0 )\r
+               {\r
+                       ofs += rv;\r
+               }\r
+               \r
+               ret = ofs;\r
+       }\r
+       else\r
+       {\r
+               ret = _SysRead(fp->FD, ptr, bytes);\r
+               if( ret == (size_t)-1)\r
+                       return -1;\r
+               if( ret == 0 && bytes > 0 ) {\r
+                       fp->Flags |= FILE_FLAG_EOF;\r
+                       return 0;\r
+               }\r
+       }\r
+\r
+       // if read was cut short        \r
+       if( ret != bytes )\r
+       {\r
+               size_t  extra = ret - (ret / size) * size;\r
+               // And it didn't fall short on a member boundary\r
+               if( extra )\r
+               {\r
+                       // Need to roll back the file pointer to the end of the last member\r
+                       _SysDebug("fread: TODO Roll back %zi bytes due to incomplete object (sz=%zi,n=%zi)",\r
+                               extra, size, num\r
+                               );\r
+               }\r
        }\r
-       ret /= size;\r
        \r
-       return ret;\r
+       return ret / size;\r
 }\r
 \r
 /**\r
@@ -665,13 +750,14 @@ EXPORT char *fgets(char *s, int size, FILE *fp)
  */\r
 EXPORT int fputc(int c, FILE *fp)\r
 {\r
-       return fwrite(&c, 1, 1, fp);\r
+       char    ch = c;\r
+       return fwrite(&ch, 1, 1, fp);\r
 }\r
 \r
 EXPORT int putchar(int c)\r
 {\r
        c &= 0xFF;\r
-       return _SysWrite(_stdout, &c, 1);\r
+       return fputc(c, stdout);\r
 }\r
 \r
 /**\r
@@ -688,20 +774,17 @@ EXPORT int fgetc(FILE *fp)
 \r
 EXPORT int getchar(void)\r
 {\r
-       char    ret = 0;\r
-       if(_SysRead(_stdin, &ret, 1) != 1)      return -1;\r
-       return ret;\r
+       return fgetc(stdin);\r
 }\r
 \r
 EXPORT int puts(const char *str)\r
 {\r
-        int    len;\r
        \r
        if(!str)        return 0;\r
-       len = strlen(str);\r
+        int    len = strlen(str);\r
        \r
-       len = _SysWrite(_stdout, str, len);\r
-       _SysWrite(_stdout, "\n", 1);\r
+       fwrite(str, 1, len, stdout);\r
+       fwrite("\n", 1, 1, stdout);\r
        return len;\r
 }\r
 \r
@@ -712,8 +795,7 @@ EXPORT int puts(const char *str)
  */\r
 FILE *get_file_struct()\r
 {\r
-        int    i;\r
-       for(i=0;i<STDIO_MAX_STREAMS;i++)\r
+       for(int i=0;i<STDIO_MAX_STREAMS;i++)\r
        {\r
                if(_iob[i].Flags & FILE_FLAG_ALLOC)\r
                        continue ;\r
index 92c598d..c56338c 100644 (file)
@@ -21,6 +21,7 @@
 
 #define FILE_FLAG_EOF          0x0100
 #define FILE_FLAG_LINEBUFFERED 0x1000  // Flush when '\n' encountered
+#define FILE_FLAG_OURBUFFER    0x2000  // Buffer is owned by stdio
 
 #define FILE_FLAG_ALLOC        0x8000  // Internal 'is used' flag
 

UCC git Repository :: git.ucc.asn.au