Logo Search packages:      
Sourcecode: scummvm version File versions

_stdio.cpp

/* ScummVM - Scumm Interpreter
 * Copyright (C) 2001  Ludvig Strigeus
 * Copyright (C) 2001-2004 The ScummVM project
 *
 * This program is free software; you can redistribute it and/or
 * modify it under the terms of the GNU General Public License
 * as published by the Free Software Foundation; either version 2
 * of the License, or (at your option) any later version.

 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.

 * You should have received a copy of the GNU General Public License
 * along with this program; if not, write to the Free Software
 * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA  02111-1307, USA.
 *
 * $Header: /cvsroot/scummvm/scummvm/backends/PalmOS/Src/missing/_stdio.cpp,v 1.11 2004/10/20 09:17:46 chrilith Exp $
 *
 */

#include <stdio.h>

FileRef     gStdioOutput = 0;

static LedProc    gStdioLedProc = NULL;
static UInt16     gStdioVolRefNum = sysInvalidRefNum;

static void dummy(Boolean){};

// TODO : implement "errno"

void StdioInit(UInt16 volRefNum, const Char *output, LedProc ledProc) {
      gStdioVolRefNum = volRefNum;
      
      if (ledProc)
            gStdioLedProc = ledProc;
      else
            gStdioLedProc = dummy;
            
      VFSFileDelete(gStdioVolRefNum, output);
      VFSFileCreate(gStdioVolRefNum, output);
      VFSFileOpen  (gStdioVolRefNum, output,vfsModeWrite, &gStdioOutput);
}

void StdioRelease() {
      VFSFileClose(gStdioOutput);
}

UInt16 fclose(FileRef *stream) {
      Err error = VFSFileClose(*stream);
      
      if (error == errNone)
            MemPtrFree(stream);

#ifdef _DEBUG_STDIO
      FrmCustomAlert(FrmWarnAlert,"error fclose",0,0);
#endif
      return error;
}

UInt16 feof(FileRef *stream) {
      Err error = VFSFileEOF(*stream);

#ifdef _DEBUG_STDIO
            switch (error)
            {
                  case vfsErrFileEOF:
                        FrmCustomAlert(FrmWarnAlert,"vfsErrFileEOF",0,0);
                        break;
                  case expErrNotOpen:
                        FrmCustomAlert(FrmWarnAlert,"expErrNotOpen",0,0);
                        break;
                  case vfsErrFileBadRef:
                        FrmCustomAlert(FrmWarnAlert,"vfsErrFileBadRef",0,0);
                        break;
                  case vfsErrIsADirectory:
                        FrmCustomAlert(FrmWarnAlert,"vfsErrIsADirectory",0,0);
                        break;
                  case vfsErrNoFileSystem:
                        FrmCustomAlert(FrmWarnAlert,"vfsErrNoFileSystem",0,0);
                        break;
            }
#endif

      return error;
}

Int16 fgetc(FileRef *stream) {
      UInt32 numBytesRead;
      Err e;
      Char c;
      
      e = VFSFileRead(*stream, 1, &c, &numBytesRead);
      return (int)(!e ? c : EOF);
}

Char *fgets(Char *s, UInt32 n, FileRef *stream) {
      UInt32 numBytesRead;
      gStdioLedProc(true);
      Err error = VFSFileRead(*stream, n, s, &numBytesRead);
      gStdioLedProc(false);
      if (error == errNone || error == vfsErrFileEOF) {
            UInt32 reset = 0;
            Char *endLine = StrChr(s, '\n');
            
            if (endLine >= s) {
                  reset = (endLine - s);
                  s[reset] = 0;
                  reset = numBytesRead - (reset + 1);
                  VFSFileSeek(*stream, vfsOriginCurrent, -reset);
            }
      
            return s;
      }
#ifdef _DEBUG_STDIO
            switch (error)
            {
                  case expErrNotOpen:
                        FrmCustomAlert(FrmWarnAlert,"expErrNotOpen",0,0);
                        break;
                  case vfsErrFileBadRef:
                        FrmCustomAlert(FrmWarnAlert,"vfsErrFileBadRef",0,0);
                        break;
                  case vfsErrFileEOF:
                        FrmCustomAlert(FrmWarnAlert,"vfsErrFileEOF",0,0);
                        break;
                  case vfsErrFilePermissionDenied:
                        FrmCustomAlert(FrmWarnAlert,"vfsErrFilePermissionDenied",0,0);
                        break;
                  case vfsErrIsADirectory:
                        FrmCustomAlert(FrmWarnAlert,"vfsErrIsADirectory",0,0);
                        break;
                  case vfsErrNoFileSystem:
                        FrmCustomAlert(FrmWarnAlert,"vfsErrNoFileSystem",0,0);
                        break;
            }
#endif

      return NULL;
}

FileRef *fopen(const Char *filename, const Char *type) {
      Err err;
      UInt16 openMode;
      FileRef *fileRefP = (FileRef *)MemPtrNew(sizeof(FileRef *));
      
      if (StrCompare(type,"r")==0)
            openMode = vfsModeRead;
      else if (StrCompare(type,"rb")==0)
            openMode = vfsModeRead;
      else if (StrCompare(type,"w")==0)
            openMode = vfsModeCreate|vfsModeWrite;
      else if (StrCompare(type,"wb")==0)
            openMode = vfsModeCreate|vfsModeWrite;
      else
            openMode = vfsModeReadWrite;

      if (openMode & vfsModeRead) {
            // if read file :
            // first try to load from the specfied card
            err = VFSFileOpen (gStdioVolRefNum, filename, openMode, fileRefP);
            //if err (not found ?) parse each avalaible card for the specified file
            if (err) {
                  UInt16 volRefNum;
                  UInt32 volIterator = vfsIteratorStart;
                  while (volIterator != vfsIteratorStop) {
                        err = VFSVolumeEnumerate(&volRefNum, &volIterator);

                        if (!err) {
                              err = VFSFileOpen (volRefNum, filename, openMode, fileRefP);
                              if (!err)
                                    return fileRefP;
                        }
                  }
            } else {
                  return fileRefP;
            }
      } else {
            // if write file :
            // use only the specified card
            err = VFSFileDelete(gStdioVolRefNum, filename); // delete it if exists
            err = VFSFileCreate(gStdioVolRefNum, filename);
            openMode = vfsModeWrite;
            if (!err) {
                  err = VFSFileOpen (gStdioVolRefNum, filename, openMode, fileRefP);
                  if (!err)
                        return fileRefP;
            }
      }

#ifdef _DEBUG_STDIO
      else
      {
            switch (err)
            {
                  case expErrCardReadOnly:
                        FrmCustomAlert(FrmWarnAlert,"expErrCardReadOnly",0,0);
                        break;
                  case expErrNotOpen:
                        FrmCustomAlert(FrmWarnAlert,"expErrNotOpen",0,0);
                        break;
                  case vfsErrBadName:
                        FrmCustomAlert(FrmWarnAlert,"vfsErrBadName",0,0);
                        break;
                  case vfsErrFileNotFound:
                        FrmCustomAlert(FrmWarnAlert,"vfsErrFileNotFound",0,0);
                        break;
                  case vfsErrFilePermissionDenied:
                        FrmCustomAlert(FrmWarnAlert,"vfsErrFilePermissionDenied",0,0);
                        break;
                  case vfsErrVolumeBadRef:
                        FrmCustomAlert(FrmWarnAlert,"vfsErrVolumeBadRef",0,0);
                        break;
                  default:
                        FrmCustomAlert(FrmWarnAlert,"unknow",0,0);
                        break;
            }
      }
#endif

      MemPtrFree(fileRefP); // prevent memory leak
      return NULL;
}

UInt32 fread(void *ptr, UInt32 size, UInt32 nitems, FileRef *stream) {
      UInt32 numBytesRead;
      gStdioLedProc(true);
      Err error = VFSFileRead(*stream, size*nitems, ptr, &numBytesRead);
      gStdioLedProc(false);
      if (error == errNone || error == vfsErrFileEOF)
            return (UInt32)(numBytesRead/size);

#ifdef _DEBUG_STDIO
            switch (error)
            {
                  case expErrNotOpen:
                        FrmCustomAlert(FrmWarn,"expErrNotOpen",0,0);
                        break;
                  case vfsErrFileBadRef:
                        FrmCustomAlert(FrmWarn,"vfsErrFileBadRef",0,0);
                        break;
                  case vfsErrFileEOF:
                        FrmCustomAlert(FrmWarn,"vfsErrFileEOF",0,0);
                        break;
                  case vfsErrFilePermissionDenied:
                        FrmCustomAlert(FrmWarn,"vfsErrFilePermissionDenied",0,0);
                        break;
                  case vfsErrIsADirectory:
                        FrmCustomAlert(FrmWarn,"vfsErrIsADirectory",0,0);
                        break;
                  case vfsErrNoFileSystem:
                        FrmCustomAlert(FrmWarn,"vfsErrNoFileSystem",0,0);
                        break;
            }
#endif
      return 0;
}

UInt32 fwrite(const void *ptr, UInt32 size, UInt32 nitems, FileRef *stream) {
      UInt32 numBytesWritten;
      gStdioLedProc(true);
      Err error = VFSFileWrite(*stream, size*nitems, ptr, &numBytesWritten);
      gStdioLedProc(false);

      if (error == errNone || error == vfsErrFileEOF)
            return (UInt32)(numBytesWritten/size);

      return NULL;
}

Int32 fseek(FileRef *stream, Int32 offset, Int32 whence) {
      Err error = VFSFileSeek(*stream, whence, offset);
      return error;
}

UInt32 ftell(FileRef *stream) {
      Err e;
      UInt32 filePos;

      e = VFSFileTell(*stream,&filePos);
      if (e != errNone)
            return e;
      
      return filePos;
}

Int32 fprintf(FileRef *stream, const Char *formatStr, ...) {
      if (!*stream)
            return 0;

      UInt32 numBytesWritten;
      Char buf[256];
      va_list va;

      va_start(va, formatStr);
      vsprintf(buf, formatStr, va);
      va_end(va);

      VFSFileWrite (*stream, StrLen(buf), buf, &numBytesWritten);
      return numBytesWritten;
}

Int32 printf(const Char *format, ...) {
      if (!*stdout)
            return 0;

      UInt32 numBytesWritten;
      Char buf[256];
      va_list va;

      va_start(va, format);
      vsprintf(buf, format, va);
      va_end(va);

      VFSFileWrite (*stdout, StrLen(buf), buf, &numBytesWritten);
      return numBytesWritten;
}

Int32 sprintf(Char* s, const Char* formatStr, ...) {
      Int16 count;
      va_list va;

      va_start(va, formatStr);
      count = vsprintf(s, formatStr, va);
      va_end(va);
      
      return count;
}

Int32 snprintf(Char* s, UInt32 len, const Char* formatStr, ...) {
      // len is ignored
      Int16 count;
      va_list va;

      va_start(va, formatStr);
      count = vsprintf(s, formatStr, va);
      va_end(va);
      
      return count;
}


/* WARNING : vsprintf
 * -------
 * This function can handle only %[+- ][.0][field length][sxXdoiucp] format strings
 * compiler option : 4byte int mode only !
 *
 * TODO : check http://www.ijs.si/software/snprintf/ for a potable implementation of vsnprintf
 * This one make use of sprintf so need to check if it works with PalmOS.
 */

static Char *StrIToBase(Char *s, Int32 i, UInt8 b) {
      const Char *conv = "0123456789ABCDEF";
      Char o;
      Int16 c, n = 0;
      Int32 div, mod;
      
      do {
            div = i / b;
            mod = i % b;
            
            s[n++]      = *(conv + mod);
            i           = div;

      } while (i >= b);

      if (i > 0) {
            s[n + 0] = *(conv + i);
            s[n + 1] = 0;
      } else {
            s[n + 0] = 0;
            n--;
      }

      for (c=0; c <= (n >> 1); c++) {
            o           = s[c];
            s[c]  = s[n - c];
            s[n - c]= o;
      }

      return s;
}

static void StrProcC_(Char *ioStr, UInt16 maxLen) {
      Char *found;
      Int16 length;
      
      while (found = StrStr(ioStr, "`c`")) {
            if (found[3] == 0) {                                  // if next char is NULL
                  length = maxLen - (found - ioStr + 2);
                  MemMove(found, found + 4, length);
                  maxLen -= 2;
            }
      }
}

static void StrProcXO(Char *ioStr, UInt16 maxLen, Char *tmp) {
      Char *found, *last, mod, fill;
      Int16 len, count, next;
      Int32 val;
            
      while (found = StrChr(ioStr, '`')) {
            last = StrChr(found + 1, '`');
            
            if (!last)
                  return;

            *last = 0;
            next  = 0;
            fill  = *(found + 1);
            mod         = *(found + 2);
            count = StrAToI(found + 3);

            len = maxLen - (last - ioStr);
            MemMove(found, (last + 1), len);

            // x and X always 8char on palmos ... o set to 8char (not supported on palmos)
            while (found[next] == '0' || found[next] == ' ')      // WARNING : reduce size only (TODO ?)
                  next++;
            
            // convert to base 8
            if (mod == 'o') {
                  StrNCopy(tmp, found + next, 8 - next);
                  tmp[8 - next]     = 0;
                  val                     = StrAToI(tmp);
                  StrIToBase(tmp, val, 8);                        // now we have the same but in octal
                  next              = 8 - StrLen(tmp);
                  MemMove(found + next, tmp, StrLen(tmp));
            } else {
                  // if val is 0, keep last 0
                  if (next == 8)
                        next = 7;
            }

            if ((8 - next) > count)
                  count = 8 - next;
                  
            if (count == 0)
                  count = 1;

            len = maxLen - (found - ioStr) - (8 - count);
            MemSet(found, next, fill);
            MemMove(found, found + (8 - count), len);

            //  ... and upper case
            if (mod == 'x') {
                  while (count--) {
                        if (*found >='A' && *found <='F')
                              *found = (*found + 32);
                        found++;
                  }
            }
      }
}

Int32 vsprintf(Char* s, const Char* formatStr, _Palm_va_list argParam) {
      Char format[256], result[256], tmp[32];
      
      Char *found, *mod, *num;
      UInt32 next;
      Boolean zero;
      Int16 count, len;
      
      MemSet(format, sizeof(format), 'x');
      MemSet(result, sizeof(result), 'y');
      MemSet(tmp, sizeof(tmp), 'z');

      StrCopy(format,formatStr);    // copy actual formatStr to temp buffer
      next = 0;                           // start of the string

      while (found = StrChr(format + next, '%')) {
            mod = found + 1;
            
            if (*mod == '%') {                  // just a % ?
                  mod++;

            } else {
                  if (*mod == '+' ||
                        *mod == '-' ||
                        *mod == ' ' )           // skip
                        mod++;
                  
                  if (*mod == '0' ||
                        *mod == '.' ) {
                        *mod++ = '0';
                        zero =  true;
                  } else {
                        zero = false;
                  }
                  
                  num = mod;
                  while (     *mod >= '0' &&
                              *mod <= '9' )     // search format char
                        mod++;
                  
                  // get the numeric value
                  if (num < mod) {
                        StrNCopy(tmp, num, mod - num);
                        tmp[mod - num] = 0;
                        count = StrAToI(tmp);
                  } else {
                        count = 0;
                  }

                  if (*mod == 'l')        // already set to %...l(x) ?
                        mod++;

                  // prepare new format
                  if (*mod == 'c') {
                        StrCopy(tmp, "`c`%c%c");

                  } else if (*mod == 'p') {
                        StrCopy(tmp, "%08lX");              // %x = %08X in palmos

                  } else {
                        len = 0;
      
                        switch (*mod) {
                              case 'x':
                              case 'X':
                              case 'o':
                                    tmp[0] = '`';
                                    tmp[1] = (zero) ? '0' : ' ';
                                    tmp[2] = *mod;    
                                    StrIToA(tmp + 3, count);
                                    len += StrLen(tmp);
                                    tmp[len++] = '`';
                                    tmp[len] = 0;
                                    
                                    if (*mod == 'o') {      // set as base 10 num and convert later
                                          *mod = 'd';
                                          count = 8;        // force 8char
                                    }

                                    break;
                        }
                              
                        StrNCopy(tmp + len, found, (num - found));
                        len += (num - found);

                        if (count) {
                              StrIToA(tmp + len, count);
                              len += StrLen(tmp + len);
                        }
                        
                        if (*mod == 'd' ||
                              *mod == 'i' ||
                              *mod == 'x' ||
                              *mod == 'X' ||
                              *mod == 'u' 
                              ) {
                              tmp[len++] = 'l';
                        }

                        tmp[len + 0] = *mod;
                        tmp[len + 1] = 0;
                  }

                  mod++;
                  MemMove(found + StrLen(tmp), mod, StrLen(mod) + 1);
                  StrNCopy(found, tmp, StrLen(tmp));
                  mod = found + StrLen(tmp);
            }
            
            next = (mod - format);
      }
      
      // Copy result in a temp buffer to process last formats
      StrVPrintF(result, format, argParam);
      StrProcC_(result, 256);
      StrProcXO(result, 256, tmp);
      StrCopy(s, result);
      
      return StrLen(s);
}

Generated by  Doxygen 1.6.0   Back to index