HP85 GPIB Disk Emulator  1.0
HP85GPIBDiskEmulator
All Data Structures Files Functions Variables Typedefs Enumerations Enumerator Macros Pages
posix.c
Go to the documentation of this file.
1 
114 #include "user_config.h"
115 
116 #include "stringsup.h"
117 #include "fatfs.h"
118 
119 #include "posix.h"
120 
121 #ifdef ESP8266
122 // FIXME ESP8266 library conflict
123 #undef strerror_r
124 #endif
125 
127 
129 int errno;
130 
139 
143 const char *sys_errlist[] =
144 {
145  "OK",
146  "Operation not permitted",
147  "No such file or directory",
148  "No such process",
149  "Interrupted system call",
150  "I/O error",
151  "No such device or address",
152  "Argument list too long",
153  "Exec format error",
154  "Bad file number",
155  "No child processes",
156  "Try again",
157  "Out of memory",
158  "Permission denied",
159  "Bad address",
160  "Block device required",
161  "Device or resource busy",
162  "File exists",
163  "Cross-device link",
164  "No such device",
165  "Not a directory",
166  "Is a directory",
167  "Invalid argument",
168  "File table overflow",
169  "Too many open files",
170  "Not a typewriter",
171  "Text file busy",
172  "File too large",
173  "No space left on device",
174  "Illegal seek",
175  "Read-only file system",
176  "Too many links",
177  "Broken pipe",
178  "Math argument out of domain of func",
179  "Math result not representable",
180  "Bad Message",
181  NULL
182 };
183 
184 // =============================================
194 MEMSPACE
195 int isatty(int fileno)
196 {
198  if(fileno >= 0 && fileno <= 2)
199  return(1);
200  return 0;
201 }
202 
203 
214 MEMSPACE
215 int
216 fgetc(FILE *stream)
217 {
218  int c;
219 
220  if(stream == NULL)
221  {
222  errno = EBADF; // Bad File Number
223  return(EOF);
224  }
225 
226 #ifdef ESP8266
227  optimistic_yield(1000);
228  wdt_reset();
229 #endif
230 
231  if ((stream->flags & __SRD) == 0)
232  return EOF;
233 
234  if ((stream->flags & __SUNGET) != 0)
235  {
236  stream->flags &= ~__SUNGET;
237  stream->len++;
238  return stream->unget;
239  }
240 
241  if (stream->flags & __SSTR)
242  {
243  c = *stream->buf;
244  if (c == '\0')
245  {
246  stream->flags |= __SEOF;
247  return EOF;
248  }
249  else
250  {
251  stream->buf++;
252  }
253  }
254  else
255  {
256  if(!stream->get)
257  {
258  printf("fgetc stream->get NULL\n");
259  return(EOF);
260  }
261 // get character from device or file
262  c = stream->get(stream);
263  if (c < 0)
264  {
265 /* if != _FDEV_ERR, assume its _FDEV_EOF */
266  stream->flags |= (c == _FDEV_ERR)? __SERR: __SEOF;
267  return EOF;
268  }
269  }
270 
271  stream->len++;
272  return (c);
273 }
274 
275 
285 MEMSPACE
286 int
287 fputc(int c, FILE *stream)
288 {
289  errno = 0;
290  int ret;
291 
292  if(stream == NULL)
293  {
294  errno = EBADF; // Bad File Number
295  return(EOF);
296  }
297 
298 #ifdef ESP8266
299  optimistic_yield(1000);
300  wdt_reset();
301 #endif
302 
303  if(stream != stdout && stream != stderr)
304  {
305  return(fatfs_putc(c,stream));
306  }
307 
308 // TTY outputs
309 
310  if ((stream->flags & __SWR) == 0)
311  return EOF;
312 
313  if (stream->flags & __SSTR)
314  {
315  if (stream->len < stream->size)
316  *stream->buf++ = c;
317  stream->len++;
318  return c;
319  }
320  else
321  {
322  if(!stream->put)
323  {
324  printf("fputc stream->put NULL\n");
325  return(EOF);
326  }
327  ret = stream->put(c, stream);
328  if(ret != EOF)
329  stream->len++;
330  return(ret);
331  }
332 }
333 
334 
336 #ifndef IO_MACROS
337 MEMSPACE
346 int
348 {
349  return(fgetc(stdin));
350 }
351 
352 
361 MEMSPACE
362 int
363 putchar(int c)
364 {
365  return(fputc(c,stdout));
366 }
367 #endif
368 
378 MEMSPACE
379 int
380 ungetc(int c, FILE *stream)
381 {
382  int fd = fileno(stream);
383  if(!isatty(fd))
384  return(EOF);
385 
386  if(c == EOF)
387  return EOF;
388  if((stream->flags & __SUNGET) != 0 )
389  return EOF;
390  if ((stream->flags & __SRD) == 0 )
391  return EOF;
392 
393  stream->flags |= __SUNGET;
394  stream->flags &= ~__SEOF;
395 
396  stream->unget = c;
397  stream->len--;
398 
399  return (c);
400 }
401 
402 
403 #ifndef IO_MACROS
404 // =============================================
412 MEMSPACE
413 int
414 putc(int c, FILE *stream)
415 {
416  return(fputc(c, stream));
417 }
418 #endif
419 
420 // =============================================
430 MEMSPACE
431 char *
432 fgets(char *str, int size, FILE *stream)
433 {
434  int c;
435  int ind = 0;
436  while(size--)
437  {
438  c = fgetc(stream);
439  if(c == EOF)
440  {
441  if( ind == 0)
442  return(NULL);
443  break;
444  }
445  if(c == '\n')
446  break;
447  if(c == 0x08)
448  {
449  if(ind > 0)
450  --ind;
451  continue;
452  }
453  str[ind++] = c;
454  }
455  str[ind] = 0;
456  return(str);
457 }
458 
459 
468 MEMSPACE
469 int
470 fputs(const char *str, FILE *stream)
471 {
472  while(*str)
473  {
474  if(fputc(*str, stream) == EOF)
475  return(EOF);
476  ++str;
477  }
478  return(0);
479 }
480 
481 
482 #ifndef IO_MACROS
483 MEMSPACE
492 int
493 puts(const char *str)
494 {
495  while(*str)
496  {
497  if(fputc(*str, stdout) == EOF)
498  return(EOF);
499  ++str;
500  }
501  return ( fputc('\n',stdout) );
502 }
503 #endif
504 
505 // =============================================
506 // =============================================
508 // =============================================
509 // =============================================
515 MEMSPACE
516 int feof(FILE *stream)
517 {
518  if(stream->flags & __SEOF)
519  return(1);
520  return(0);
521 }
522 
523 
533 MEMSPACE
534 int fgetpos(FILE *stream, size_t *pos)
535 {
536  long offset = ftell(stream);
537  *pos = offset;
538  if(offset == -1)
539  return(-1);
540  return( 0 );
541 }
542 
543 
557 MEMSPACE
558 int fseek(FILE *stream, long offset, int whence)
559 {
560  long ret;
561 
562  int fn = fileno(stream);
563  if(fn < 0)
564  return(-1);
565 
566  ret = lseek(fn, offset, whence);
567 
568  if(ret == -1)
569  return(-1);
570 
571  return(0);
572 }
573 
574 
584 MEMSPACE
585 int fsetpos(FILE *stream, size_t *pos)
586 {
587  return (fseek(stream, (size_t) *pos, SEEK_SET) );
588 }
589 
590 
599 MEMSPACE
600 long ftell(FILE *stream)
601 {
602  errno = 0;
603 
604  int fn = fileno(stream);
605  if(isatty(fn))
606  return(-1);
607 // fileno_to_fatfs checks for fd out of bounds
608  FIL *fh = fileno_to_fatfs(fn);
609  if ( fh == NULL )
610  {
611  errno = EBADF;
612  return(-1);
613  }
614 
615  return( fh->fptr );
616 }
617 
618 
632 MEMSPACE
633 off_t lseek(int fileno, off_t position, int whence)
634 {
635  FRESULT res;
636  FIL *fh;
637  errno = 0;
638  FILE *stream;
639 
640 // fileno_to_fatfs checks for fd out of bounds
641  fh = fileno_to_fatfs(fileno);
642  if(fh == NULL)
643  {
644  errno = EMFILE;
645  return(-1);
646  }
647  if(isatty(fileno))
648  return(-1);
649 
650  stream = fileno_to_stream(fileno);
651  stream->flags |= __SUNGET;
652 
653  if(whence == SEEK_END)
654  position += f_size(fh);
655  else if(whence==SEEK_CUR)
656  position += fh->fptr;
657 
658  res = f_lseek(fh, position);
659  if(res)
660  {
661  errno = fatfs_to_errno(res);
662  return -1;
663  }
664  if(position != f_tell(fh) )
665  {
666  printf("Seek failed");
667  return(-1L);
668  }
669  return (fh->fptr);
670 }
671 
672 
680 MEMSPACE
681 void rewind( FILE *stream)
682 {
683  fseek(stream, 0L, SEEK_SET);
684 }
685 
686 
687 // =============================================
688 // =============================================
690 // =============================================
691 // =============================================
700 MEMSPACE
701 int close(int fileno)
702 {
703  FILE *stream;
704  FIL *fh;
705  int res;
706 
707  errno = 0;
708 
709 // checks if fileno out of bounds
710  stream = fileno_to_stream(fileno);
711  if(stream == NULL)
712  {
713  return(-1);
714  }
715 
716 // fileno_to_fatfs checks for fileno out of bounds
717  fh = fileno_to_fatfs(fileno);
718  if(fh == NULL)
719  {
720  return(-1);
721  }
722  res = f_close(fh);
724  if (res != FR_OK)
725  {
726  errno = fatfs_to_errno(res);
727  return(-1);
728  }
729  return(0);
730 }
731 
732 
740 MEMSPACE
741 int fileno(FILE *stream)
742 {
743  int fileno;
744 
745  if(stream == NULL)
746  {
747  errno = EBADF;
748  return(-1);
749  }
750 
751  for(fileno=0; fileno<MAX_FILES; ++fileno)
752  {
753  if ( __iob[fileno] == stream)
754  return(fileno);
755  }
756  return(-1);
757 }
758 
759 
771 MEMSPACE
773 {
774  FILE *stream;
775  if(fileno < 0 || fileno >= MAX_FILES)
776  {
777  errno = EBADF;
778  return(NULL);
779  }
780 
781  stream = __iob[fileno];
782  if(stream == NULL)
783  {
784  errno = EBADF;
785  return(NULL);
786  }
787  return(stream);
788 }
789 
790 
800 MEMSPACE
801 FILE *fopen(const char *path, const char *mode)
802 {
803  int flags = posix_fopen_modes_to_open(mode);
804  int fileno = open(path, flags);
805 
806 // checks if fileno out of bounds
807  return( fileno_to_stream(fileno) );
808 }
809 
810 
822 MEMSPACE
823 size_t fread(void *ptr, size_t size, size_t nmemb, FILE *stream)
824 {
825  size_t count = size * nmemb;
826  int fn = fileno(stream);
827  ssize_t ret;
828 
829 // read() checks for fn out of bounds
830  ret = read(fn, ptr, count);
831  if(ret < 0)
832  return(0);
833 
834  return((size_t) ret);
835 }
836 
837 
847 MEMSPACE
848 int ftruncate(int fd, off_t length)
849 {
850  errno = 0;
851  FIL *fh;
852  FRESULT rc;
853 
854  if(isatty(fd))
855  return(-1);
856 // fileno_to_fatfs checks for fd out of bounds
857  fh = fileno_to_fatfs(fd);
858  if(fh == NULL)
859  {
860  return(-1);
861  }
862  rc = f_lseek(fh, length);
863  if (rc != FR_OK)
864  {
865  errno = fatfs_to_errno(rc);
866  return(-1);
867  }
868  rc = f_truncate(fh);
869  if (rc != FR_OK)
870  {
871  errno = fatfs_to_errno(rc);
872  return(-1);
873  }
874  return(0);
875 }
876 
877 
889 MEMSPACE
890 size_t fwrite(const void *ptr, size_t size, size_t nmemb, FILE *stream)
891 {
892  size_t count = size * nmemb;
893  int fn = fileno(stream);
894  ssize_t ret;
895 
896 // write () checks for fn out of bounds
897  ret = write(fn, ptr, count);
898 
899  if(ret < 0)
900  return(0);
901 
902  return((size_t) ret);
903 }
904 
905 
915 MEMSPACE
916 int open(const char *pathname, int flags)
917 {
918  int fileno;
919  int fatfs_modes;
920  FILE *stream;
921  FIL *fh;
922  int res;
923 
924  errno = 0;
925 
926 // FIXME Assume here that the disk interface mmc_init was already called
927 #if 0
928 // Checks Disk status
929  res = mmc_init(0);
930 
931  if(res != RES_OK)
932  {
933  errno = fatfs_to_errno(res);
934  return(-1);
935  }
936 #endif
937 
938  if((flags & O_ACCMODE) == O_RDWR)
939  fatfs_modes = FA_READ | FA_WRITE;
940  else if((flags & O_ACCMODE) == O_RDONLY)
941  fatfs_modes = FA_READ;
942  else
943  fatfs_modes = FA_WRITE;
944 
945  if(flags & O_CREAT)
946  {
947  if(flags & O_TRUNC)
948  fatfs_modes |= FA_CREATE_ALWAYS;
949  else
950  fatfs_modes |= FA_OPEN_ALWAYS;
951  }
952 
954 
955 // checks if fileno out of bounds
956  stream = fileno_to_stream(fileno);
957  if(stream == NULL)
958  {
960  return(-1);
961  }
962 
963 // fileno_to_fatfs checks for fileno out of bounds
964  fh = fileno_to_fatfs(fileno);
965  if(fh == NULL)
966  {
968  errno = EBADF;
969  return(-1);
970  }
971  res = f_open(fh, pathname, (BYTE) (fatfs_modes & 0xff));
972  if(res != FR_OK)
973  {
974  errno = fatfs_to_errno(res);
976  return(-1);
977  }
978  if(flags & O_APPEND)
979  {
981  res = f_lseek(fh, f_size(fh));
982  if (res != FR_OK)
983  {
984  errno = fatfs_to_errno(res);
985  f_close(fh);
987  return(-1);
988  }
989  }
990 
991  if((flags & O_ACCMODE) == O_RDWR)
992  {
993 // FIXME fdevopen should do this
994  stream->put = fatfs_putc;
995  stream->get = fatfs_getc;
996  stream->flags = _FDEV_SETUP_RW;
997  }
998  else if((flags & O_ACCMODE) == O_RDONLY)
999  {
1000 // FIXME fdevopen should do this
1001  stream->put = NULL;
1002  stream->get = fatfs_getc;
1003  stream->flags = _FDEV_SETUP_READ;
1004  }
1005  else
1006  {
1007 // FIXME fdevopen should do this
1008  stream->put = fatfs_putc;
1009  stream->get = NULL;
1010  stream->flags = _FDEV_SETUP_WRITE;
1011  }
1012 
1013  return(fileno);
1014 }
1015 
1016 
1027 MEMSPACE
1028 ssize_t read(int fd, const void *buf, size_t count)
1029 {
1030  UINT size;
1031  UINT bytes = count;
1032  int res;
1033  int ret;
1034  FIL *fh;
1035  FILE *stream;
1036 
1037 //FIXME
1038  *(char *) buf = 0;
1039 
1040  errno = 0;
1041 
1042 // TTY read function
1043 // FIXME should we really be blocking ???
1044  stream = fileno_to_stream(fd);
1045  if(stream == stdin)
1046  {
1047  char *ptr = (char *) buf;
1048 // ungetc is undefined for read
1049  stream->flags |= __SUNGET;
1050  size = 0;
1051  while(count--)
1052  {
1053  ret = fgetc(stream);
1054  if(ret < 0)
1055  break;
1056 
1057  *ptr++ = ret;
1058  ++size;
1059  }
1060  return(size);
1061  }
1062  if(stream == stdout || stream == stderr)
1063  {
1064  return(-1);
1065  }
1066 
1067 // fileno_to_fatfs checks for fd out of bounds
1068  fh = fileno_to_fatfs(fd);
1069  if ( fh == NULL )
1070  {
1071  errno = EBADF;
1072  return(-1);
1073  }
1074 
1075  res = f_read(fh, (void *) buf, bytes, &size);
1076  if(res != FR_OK)
1077  {
1078  errno = fatfs_to_errno(res);
1079  return(-1);
1080  }
1081  return ((ssize_t) size);
1082 }
1083 
1084 
1090 MEMSPACE
1091 void sync(void)
1092 {
1093  FIL *fh;
1094  int i;
1095 
1096  for(i=0;i<MAX_FILES;++i)
1097  {
1098  if(isatty(i))
1099  continue;
1100 
1101 // fileno_to_fatfs checks for i out of bounds
1102  fh = fileno_to_fatfs(i);
1103  if(fh == NULL)
1104  continue;
1105 
1106  (void ) syncfs(i);
1107  }
1108 }
1109 
1110 
1118 MEMSPACE
1119 int syncfs(int fd)
1120 {
1121  FIL *fh;
1122  FRESULT res;
1123  FILE *stream;
1124 
1125  errno = 0;
1126 
1127  if(isatty(fd))
1128  {
1129  errno = EBADF;
1130  return(-1);
1131  }
1132  stream = fileno_to_stream(fd);
1133 // reset unget on sync
1134  stream->flags |= __SUNGET;
1135 
1136 // fileno_to_fatfs checks for fd out of bounds
1137  fh = fileno_to_fatfs(fd);
1138  if(fh == NULL)
1139  {
1140  errno = EBADF;
1141  return(-1);
1142  }
1143 
1144  res = f_sync ( fh );
1145  if (res != FR_OK)
1146  {
1147  errno = fatfs_to_errno(res);
1148  return(-1);
1149  }
1150  return(0);
1151 }
1152 
1153 
1163 MEMSPACE
1164 int truncate(const char *path, off_t length)
1165 {
1166  errno = 0;
1167  FIL fh;
1168  FRESULT rc;
1169 
1170  rc = f_open(&fh , path, FA_OPEN_EXISTING | FA_READ | FA_WRITE);
1171  if (rc != FR_OK)
1172  {
1173  errno = fatfs_to_errno(rc);
1174  return(-1);
1175  }
1176  rc = f_lseek(&fh, length);
1177  if (rc != FR_OK)
1178  {
1179  errno = fatfs_to_errno(rc);
1180  return(-1);
1181  }
1182  rc = f_truncate(&fh);
1183  if (rc != FR_OK)
1184  {
1185  errno = fatfs_to_errno(rc);
1186  return(-1);
1187  }
1188  return(0);
1189 }
1190 
1191 
1201 MEMSPACE
1202 ssize_t write(int fd, const void *buf, size_t count)
1203 {
1204  UINT size;
1205  UINT bytes = count;
1206  FRESULT res;
1207  FIL *fh;
1208  FILE *stream;
1209  errno = 0;
1210 
1211 // TTY read function
1212  stream = fileno_to_stream(fd);
1213  if(stream == stdout || stream == stderr)
1214  {
1215  char *ptr = (char *) buf;
1216  size = 0;
1217  while(count--)
1218  {
1219  int c,ret;
1220  c = *ptr++;
1221  ret = fputc(c, stream);
1222  if(c != ret)
1223  break;
1224 
1225  ++size;
1226  }
1227  return(size);
1228  }
1229  if(stream == stdin)
1230  {
1231  return(-1);
1232  }
1233 
1234 // fileno_to_fatfs checks for fd out of bounds
1235  fh = fileno_to_fatfs(fd);
1236  if ( fh == NULL )
1237  {
1238  errno = EBADF;
1239  return(-1);
1240  }
1241 
1242  res = f_write(fh, buf, bytes, &size);
1243  if(res != FR_OK)
1244  {
1245  errno = fatfs_to_errno(res);
1246  return(-1);
1247  }
1248  return ((ssize_t) size);
1249 }
1250 
1251 
1257 
1260 MEMSPACE
1261 int fclose(FILE *stream)
1262 {
1263  int fn = fileno(stream);
1264  if(fn < 0)
1265  return(EOF);
1266 
1267  return( close(fn) );
1268 }
1269 
1270 
1271 // =============================================
1272 // =============================================
1274 // =============================================
1275 // =============================================
1276 
1282 MEMSPACE
1283 void dump_stat(struct stat *sp)
1284 {
1285  mode_t mode = sp->st_mode;
1286 
1287  printf("\tSize: %lu\n", (uint32_t)sp->st_size);
1288 
1289  printf("\tType: ");
1290  if(S_ISDIR(mode))
1291  printf("DIR\n");
1292  else if(S_ISREG(mode))
1293  printf("File\n");
1294  else
1295  printf("Unknown\n");
1296 
1297  printf("\tMode: %lo\n", (uint32_t)sp->st_mode);
1298  printf("\tUID: %lu\n", (uint32_t)sp->st_uid);
1299  printf("\tGID: %lu\n", (uint32_t)sp->st_gid);
1300  printf("\tatime: %s\n",mctime((time_t)sp->st_atime));
1301  printf("\tmtime: %s\n",mctime((time_t)sp->st_mtime));
1302  printf("\tctime: %s\n",mctime((time_t)sp->st_ctime));
1303 }
1304 
1305 
1306 #if 0
1307 MEMSPACE
1322 int fstat(int fd, struct stat *buf)
1323 {
1324  FIL *fh;
1325  FRESULT rc;
1326 
1327  if(isatty(fd))
1328  return(-1);
1329 
1330 //FIXME TODO
1331  return(-1);
1332 
1333 }
1334 #endif
1335 
1345 MEMSPACE
1346 char *mctime(time_t timev)
1347 {
1348  errno = 0;
1349  char *ptr = (char *)ctime_gm(&timev);
1350  int len;
1351  len = strlen(ptr);
1352  if(len && ptr[len-1] == '\n')
1353  ptr[len-1] = 0;
1354  return(ptr);
1355 }
1356 
1357 
1367 MEMSPACE
1368 int stat(char *name, struct stat *buf)
1369 {
1370  FILINFO info;
1371  int res;
1372  time_t epoch;
1373  uint16_t mode;
1374  errno = 0;
1375 
1376 // f_stat does not handle / \ or . as root directory
1377  if (MATCH(name,"\\") || MATCH(name,"/") || MATCH(name,".") )
1378  {
1379  buf->st_atime = 0;
1380  buf->st_mtime = 0;
1381  buf->st_ctime = 0;
1382  buf->st_uid= 0;
1383  buf->st_gid= 0;
1384  buf->st_size = 0;
1385  buf->st_mode = S_IFDIR;
1386  return(0);
1387  }
1388 
1389  res = f_stat(name, &info);
1390  if(res != FR_OK)
1391  {
1392  errno = fatfs_to_errno(res);
1393  return(-1);
1394  }
1395 
1396  buf->st_size = info.fsize;
1397  epoch = fat_time_to_unix(info.fdate, info.ftime);
1398  buf->st_atime = epoch; // Access time
1399  buf->st_mtime = epoch; // Modification time
1400  buf->st_ctime = epoch; // Creation time
1401 
1402 // We only handle read only case
1403  mode = (FATFS_R | FATFS_X);
1404  if( !(info.fattrib & AM_RDO))
1405  mode |= (FATFS_W); // enable write if NOT read only
1406 
1407  if(info.fattrib & AM_SYS)
1408  {
1409  buf->st_uid= 0;
1410  buf->st_gid= 0;
1411  }
1412  {
1413  buf->st_uid=1000;
1414  buf->st_gid=1000;
1415  }
1416 
1417  if(info.fattrib & AM_DIR)
1418  mode |= S_IFDIR;
1419  else
1420  mode |= S_IFREG;
1421  buf->st_mode = mode;
1422 
1423  return(0);
1424 }
1425 
1426 
1431 MEMSPACE
1432 int utime(const char *filename, const struct utimbuf *times)
1433 {
1434 
1435  FILINFO fno;
1436  uint16_t fdate,ftime;
1437  time_t ut;
1438  int res;
1439 
1440  if(times)
1441  ut = times->modtime;
1442  else
1443  ut = time(0);
1444 
1445  unix_time_to_fat(ut, (uint16_t *) &fdate, (uint16_t *) &ftime);
1446 
1447  fno.fdate = fdate;
1448  fno.ftime = ftime;
1449 
1450  res = f_utime(filename, (FILINFO *) &fno);
1451 
1452  return( fatfs_to_errno(res) );
1453 }
1454 
1455 
1456 // =============================================
1457 // =============================================
1459 // =============================================
1460 // =============================================
1468 MEMSPACE
1469 char *basename(char *str)
1470 {
1471  char *base = str;
1472  if(!str)
1473  return("");
1474  while(*str)
1475  {
1476  if(*str++ == '/')
1477  base = str;
1478  }
1479  return(base);
1480 }
1481 
1482 
1489 MEMSPACE
1490 char *baseext(char *str)
1491 {
1492  char *ext = "";
1493 
1494  while(*str)
1495  {
1496  if(*str++ == '.')
1497  ext = str;
1498  }
1499  return(ext);
1500 }
1501 
1502 
1511 MEMSPACE
1512 int chdir(const char *pathname)
1513 {
1514  errno = 0;
1515 
1516  int res = f_chdir(pathname);
1517  if(res != FR_OK)
1518  {
1519  errno = fatfs_to_errno(res);
1520  return(-1);
1521  }
1522  return(0);
1523 }
1524 
1525 
1537 MEMSPACE
1538 int chmod(const char *pathname, mode_t mode)
1539 {
1540  int rc;
1541  errno = 0;
1542 
1543 // FIXME for now we combine user,group and other perms and ask if anyone has write perms ?
1544 
1545 // Read only ???
1546  if ( !( mode & ( S_IWUSR | S_IWGRP | S_IWOTH)))
1547  {
1548 // file is read only
1549  rc = f_chmod(pathname, AM_RDO, AM_RDO);
1550  if (rc != FR_OK)
1551  {
1552  errno = fatfs_to_errno(rc);
1553  return(-1);
1554  }
1555  }
1556 
1557  return(0);
1558 }
1559 
1560 
1576 MEMSPACE
1577 int dirname(char *str)
1578 {
1579  int end = 0;
1580  int ind = 0;
1581 
1582  if(!str)
1583  return(0);
1584 
1585  while(*str)
1586  {
1587  if(*str == '/')
1588  end = ind;
1589  ++str;
1590  ++ind;
1591  }
1592  return(end);
1593 }
1594 
1595 
1596 #if 0
1597 MEMSPACE
1607 int fchmod(int fd, mode_t mode)
1608 {
1609 //FIXME TODO
1610  return (-1);
1611 }
1612 #endif
1613 
1622 MEMSPACE
1623 char *getcwd(char *pathname, int len)
1624 {
1625  int res;
1626  errno = 0;
1627 
1628  res = f_getcwd(pathname, len);
1629  if(res != FR_OK)
1630  {
1631  errno = fatfs_to_errno(res);
1632  return(NULL);
1633  }
1634  return(pathname);
1635 }
1636 
1637 
1646 MEMSPACE
1647 int mkdir(const char *pathname, mode_t mode)
1648 {
1649  errno = 0;
1650 
1651  if(mode)
1652  {
1653  if(chmod(pathname, mode))
1654  return(-1);
1655  }
1656 
1657  int res = f_mkdir(pathname);
1658  if(res != FR_OK)
1659  {
1660  errno = fatfs_to_errno(res);
1661  return(-1);
1662  }
1663  return(0);
1664 }
1665 
1666 
1676 MEMSPACE
1677 int rename(const char *oldpath, const char *newpath)
1678 {
1679 /* Rename an object */
1680  int rc;
1681  errno = 0;
1682  rc = f_rename(oldpath, newpath);
1683  if(rc)
1684  {
1685  errno = fatfs_to_errno(rc);
1686  return(-1);
1687  }
1688  return(0);
1689 }
1690 
1691 
1700 MEMSPACE
1701 int rmdir(const char *pathname)
1702 {
1703  errno = 0;
1704  int res = f_unlink(pathname);
1705  if(res != FR_OK)
1706  {
1707  errno = fatfs_to_errno(res);
1708  return(-1);
1709  }
1710  return(0);
1711 }
1712 
1713 
1722 MEMSPACE
1723 int unlink(const char *pathname)
1724 {
1725  errno = 0;
1726  int res = f_unlink(pathname);
1727  if(res != FR_OK)
1728  {
1729  errno = fatfs_to_errno(res);
1730  return(-1);
1731  }
1732  return(0);
1733 }
1734 
1735 
1736 // =============================================
1737 // =============================================
1739 // =============================================
1740 // =============================================
1748 int closedir(DIR *dirp)
1749 {
1750  int res = f_closedir (dirp);
1751  if(res != FR_OK)
1752  {
1753  errno = fatfs_to_errno(res);
1754  return(-1);
1755  }
1756  return(0);
1757 }
1758 
1759 
1767 static DIR _dp;
1768 DIR *opendir(const char *pathdir)
1769 {
1770  int res = f_opendir((DIR *) &_dp, pathdir);
1771  if(res != FR_OK)
1772  {
1773  errno = fatfs_to_errno(res);
1774  return(NULL);
1775  }
1776  return ((DIR *) &_dp);
1777 }
1778 
1779 
1787 static dirent_t _de;
1789 {
1790  FILINFO fno;
1791  int len;
1792  int res;
1793 
1794  _de.d_name[0] = 0;
1795  res = f_readdir ( dirp, &fno );
1796  if(res != FR_OK)
1797  {
1798  errno = fatfs_to_errno(res);
1799  return(NULL);
1800  }
1801  len = strlen(fno.fname);
1802  strncpy(_de.d_name,fno.fname,len);
1803  _de.d_name[len] = 0;
1804  return( (dirent_t *) &_de);
1805 }
1806 
1807 
1808 // =============================================
1809 // =============================================
1811 // =============================================
1812 // =============================================
1813 
1819 MEMSPACE
1820 void clrerror(FILE *stream)
1821 {
1822  stream->flags &= ~__SEOF;
1823  stream->flags &= ~__SERR;
1824 }
1825 
1826 
1832 MEMSPACE
1833 int ferror(FILE *stream)
1834 {
1835  if(stream->flags & __SERR)
1836  return(1);
1837  return(0);
1838 }
1839 
1840 
1849 MEMSPACE
1850 void perror(const char *s)
1851 {
1852  const char *ptr = NULL;
1853 
1854  if(errno >=0 && errno < EBADMSG)
1855  ptr = sys_errlist[errno];
1856  else
1857  ptr = sys_errlist[EBADMSG];
1858 
1859  if(s && *s)
1860  printf("%s: %s\n", s, ptr);
1861  else
1862  printf("%s\n", ptr);
1863 }
1864 
1865 
1874 MEMSPACE
1875 char
1876 WEAK_ATR *strerror(int errnum)
1877 {
1878  return( (char *)sys_errlist[errnum] );
1879 }
1880 
1881 
1892 MEMSPACE
1893 char *strerror_r(int errnum, char *buf, size_t buflen)
1894 {
1895  strncpy(buf, sys_errlist[errnum], buflen);
1896  return(buf);
1897 }
1898 
1899 
1900 // =============================================
1901 // =============================================
1903 // =============================================
1904 // =============================================
1910 MEMSPACE
1911 FILE *
1912 fdevopen(int (*put)(char, FILE *), int (*get)(FILE *))
1913 {
1914  FILE *s;
1915 
1916  if (put == 0 && get == 0)
1917  return 0;
1918 
1919  if ((s = safecalloc(1, sizeof(FILE))) == 0)
1920  return 0;
1921 
1922  s->flags = __SMALLOC;
1923 
1924  if (get != 0)
1925  {
1926  s->get = get;
1927  s->flags |= __SRD;
1928 // We assign the first device with a read discriptor to stdin
1929 // Only assign once
1930  if (stdin == 0)
1931  stdin = s;
1932  }
1933 
1934  if (put != 0)
1935  {
1936  s->put = put;
1937  s->flags |= __SWR;
1938 // NOTE: We assign the first device with a write to both STDOUT andd STDERR
1939 
1940 // Only assign in unassigned
1941  if (stdout == 0)
1942  stdout = s;
1943  if (stderr == 0)
1944  stderr = s;
1945  }
1946 
1947  return s;
1948 }
1949 
1950 
1951 // =============================================
1952 // =============================================
1954 // =============================================
1955 // =============================================
1956 
1960 int mkfs(char *name)
1961 {
1962  FATFS fs;
1963  uint8_t *mem;
1964  int res;
1965  int len;
1966  int c;
1967  char dev[4];
1968  static const MKFS_PARM defopt = /* Default parameter */
1969  {
1970  FM_FAT32, 0, 0, 2, 0
1971  };
1972  const MKFS_PARM *opt = &defopt;
1973 
1974  len = MATCH(name,"/dev/sd");
1975  if(!len)
1976  {
1977  printf("mkfs expected /dev/sda .. /dev/sdj\n");
1978  return(0);
1979  }
1980 // Convert /dev/sd[a-j] to 0: .. 9:
1981  dev[1] = ':';
1982  dev[2] = 0;
1983  c = tolower( name[len-1] );
1984  if(c >= 'a' && c <= ('a' + 9))
1985  dev[0] = (c - 'a');
1986  dev[3] = 0;
1987 
1988 // Register work area to the logical drive 0:
1989  res = f_mount(&fs, dev, 0);
1990  if(!res)
1991  {
1992  printf("mkfs f_mount failed\n");
1993  return(0);
1994  }
1995 
1996 // Allocate memory for mkfs function
1997  mem = safemalloc(1024);
1998  if(!mem)
1999  {
2000  printf("mkfs calloc failed\n");
2001  return(0);
2002  }
2003 
2004 // Create FAT volume on the logical drive 0
2005 // 2nd argument is ignored. */
2006  res = f_mkfs(dev, opt, mem, 1024);
2007  if(res)
2008  {
2009  printf("mkfs f_mkfs failed\n");
2010  safefree(mem);
2011  return(0);
2012  }
2013  safefree(mem);
2014  return(1);
2015 }
2016 
2017 
2030 MEMSPACE
2031 int fatfs_getc(FILE *stream)
2032 {
2033  FIL *fh;
2034  UINT size;
2035  int res;
2036  uint8_t c;
2037  long pos;
2038 
2039  errno = 0;
2040 
2041  if(stream == NULL)
2042  {
2043  errno = EBADF; // Bad File Number
2044  return(EOF);
2045  }
2046 
2047  fh = (FIL *) fdev_get_udata(stream);
2048  if(fh == NULL)
2049  {
2050  errno = EBADF; // Bad File Number
2051  return(EOF);
2052  }
2053 
2054  res = f_read(fh, &c, 1, (UINT *) &size);
2055  if( res != FR_OK || size != 1)
2056  {
2057  errno = fatfs_to_errno(res);
2058  stream->flags |= __SEOF;
2059  return(EOF);
2060  }
2061 
2062 // AUTOMATIC end of line METHOD detection
2063 // ALWAYS return '\n' for ALL methods
2064 // History: End of line (EOL) characters sometimes differ, mostly legacy systems, and modern UNIX (which uses just '\n')
2065 // '\r' ONLY
2066 // '\r\n'
2067 // '\n'
2068 // The difference was mostly from the way old mechanical printers were controlled.
2069 // '\n' (New Line = NL) advanced the line
2070 // '\r' (Charage Return = CR) moved the print head to start of line
2071 // '\t' (Tabstop = TAB)
2072 // '\f' (Form feed = FF)
2073 // The problem with mechanical devices is that each had differing control and time delays to deal with.
2074 // (TAB, CR, NL and FF) did different things and took differing times depending on the device.
2075 //
2076 // Long before DOS UNIX took the position that controlling physical devices must be a device drivers problem only.
2077 // They reasoned if users had to worry about all the ugly controll and timing issues no code would be portable.
2078 // Therefore they made NL just a SYMBOL for the driver to determine what to do.
2079 // This design philosophy argued if you needed better control its better to use a real designed purposed tool for it.
2080 // (ie. like curses or termcap).
2081 
2082 // Here to deal with those other old ugly stupid pointless EOL methods we convert to just a symbol '\n'
2083 // FROM '\n' OR '\r'char OR '\r\n' TO '\n'
2084 // Note: char != '\n'
2085  if(c == '\r')
2086  {
2087 // PEEK forward 1 character
2088  pos = f_tell(fh);
2089 // Check for trailing '\n' or EOF
2090  res = f_read(fh, &c, 1, (UINT *) &size);
2091  if(res != FR_OK || size != 1)
2092  {
2093 // '\r' with EOF impiles '\n'
2094  return('\n');
2095  }
2096 // This file must be '\r' ONLY for end of line
2097  if(c != '\n')
2098  {
2099 // Not '\n' or EOF o move file pointer back to just after the '\r'
2100  f_lseek(fh, pos);
2101  return('\n');
2102  }
2103  c = '\n';
2104  }
2105  return(c & 0xff);
2106 }
2107 
2108 
2121 MEMSPACE
2122 int fatfs_putc(char c, FILE *stream)
2123 {
2124  int res;
2125  FIL *fh;
2126  UINT size;
2127 
2128  errno = 0;
2129  if(stream == NULL)
2130  {
2131  errno = EBADF; // Bad File Number
2132  return(EOF);
2133  }
2134 
2135  fh = (FIL *) fdev_get_udata(stream);
2136  if(fh == NULL)
2137  {
2138  errno = EBADF; // Bad File Number
2139  return(EOF);
2140  }
2141 
2142  res = f_write(fh, &c, 1, (UINT *) &size);
2143  if( res != FR_OK || size != 1)
2144  {
2145  errno = fatfs_to_errno(res);
2146  stream->flags |= __SEOF;
2147  return(EOF);
2148  }
2149  return(c);
2150 }
2151 
2152 
2162 MEMSPACE
2164 {
2165  switch( Result )
2166  {
2167  case FR_OK: /* FatFS (0) Succeeded */
2168  return (0); /* POSIX OK */
2169  case FR_DISK_ERR: /* FatFS (1) A hard error occurred in the low level disk I/O layer */
2170  return (EIO); /* POSIX Input/output error (POSIX.1) */
2171 
2172  case FR_INT_ERR: /* FatFS (2) Assertion failed */
2173  return (EPERM); /* POSIX Operation not permitted (POSIX.1) */
2174 
2175  case FR_NOT_READY: /* FatFS (3) The physical drive cannot work */
2176  return (EBUSY); /* POSIX Device or resource busy (POSIX.1) */
2177 
2178  case FR_NO_FILE: /* FatFS (4) Could not find the file */
2179  return (ENOENT); /* POSIX No such file or directory (POSIX.1) */
2180 
2181  case FR_NO_PATH: /* FatFS (5) Could not find the path */
2182  return (ENOENT); /* POSIX No such file or directory (POSIX.1) */
2183 
2184  case FR_INVALID_NAME: /* FatFS (6) The path name format is invalid */
2185  return (EINVAL); /* POSIX Invalid argument (POSIX.1) */
2186 
2187  case FR_DENIED: /* FatFS (7) Access denied due to prohibited access or directory full */
2188  return (EACCES); /* POSIX Permission denied (POSIX.1) */
2189  case FR_EXIST: /* FatFS (8) Access denied due to prohibited access */
2190  return (EACCES); /* POSIX Permission denied (POSIX.1) */
2191 
2192  case FR_INVALID_OBJECT: /* FatFS (9) The file/directory object is invalid */
2193  return (EINVAL); /* POSIX Invalid argument (POSIX.1) */
2194 
2195  case FR_WRITE_PROTECTED: /* FatFS (10) The physical drive is write protected */
2196  return(EROFS); /* POSIX Read-only filesystem (POSIX.1) */
2197 
2198  case FR_INVALID_DRIVE: /* FatFS (11) The logical drive number is invalid */
2199  return(ENXIO); /* POSIX No such device or address (POSIX.1) */
2200 
2201  case FR_NOT_ENABLED: /* FatFS (12) The volume has no work area */
2202  return (ENOSPC); /* POSIX No space left on device (POSIX.1) */
2203 
2204  case FR_NO_FILESYSTEM: /* FatFS (13) There is no valid FAT volume */
2205  return(ENXIO); /* POSIX No such device or address (POSIX.1) */
2206 
2207  case FR_MKFS_ABORTED: /* FatFS (14) The f_mkfs() aborted due to any parameter error */
2208  return (EINVAL); /* POSIX Invalid argument (POSIX.1) */
2209 
2210  case FR_TIMEOUT: /* FatFS (15) Could not get a grant to access the volume within defined period */
2211  return (EBUSY); /* POSIX Device or resource busy (POSIX.1) */
2212 
2213  case FR_LOCKED: /* FatFS (16) The operation is rejected according to the file sharing policy */
2214  return (EBUSY); /* POSIX Device or resource busy (POSIX.1) */
2215 
2216  case FR_NOT_ENOUGH_CORE: /* FatFS (17) LFN working buffer could not be allocated */
2217  return (ENOMEM); /* POSIX Not enough space (POSIX.1) */
2218 
2219  case FR_TOO_MANY_OPEN_FILES: /* FatFS (18) Number of open files > _FS_SHARE */
2220  return (EMFILE); /* POSIX Too many open files (POSIX.1) */
2221 
2222  case FR_INVALID_PARAMETER: /* FatFS (19) Given parameter is invalid */
2223  return (EINVAL); /* POSIX Invalid argument (POSIX.1) */
2224 
2225  }
2226  return (EBADMSG); /* POSIX Bad message (POSIX.1) */
2227 }
2228 
2229 
2237 MEMSPACE
2239 {
2240  int i;
2241 
2242  FILE *stream;
2243 
2244  if(fh == NULL)
2245  {
2246  errno = EBADF;
2247  return(-1);
2248  }
2249 
2250  for(i=0;i<MAX_FILES;++i)
2251  {
2252  stream = __iob[i];
2253  if(stream)
2254  {
2255  if( fh == (FIL *) fdev_get_udata(stream) )
2256  return(i);
2257  }
2258  }
2259  errno = EBADF;
2260  return(-1);
2261 }
2262 
2263 
2275 MEMSPACE
2276 time_t fat_time_to_unix(uint16_t date, uint16_t time)
2277 {
2278  struct tm tp;
2279  time_t unix;
2280 
2281  memset(&tp, 0, sizeof(struct tm));
2282 
2283  tp.tm_sec = (time << 1) & 0x3e; // 2 second resolution
2284  tp.tm_min = ((time >> 5) & 0x3f);
2285  tp.tm_hour = ((time >> 11) & 0x1f);
2286  tp.tm_mday = (date & 0x1f);
2287  tp.tm_mon = ((date >> 5) & 0x0f) - 1;
2288  tp.tm_year = ((date >> 9) & 0x7f) + 80;
2289  unix = timegm( &tp );
2290  return( unix );
2291 }
2292 
2293 
2301 MEMSPACE
2302 void unix_time_to_fat(time_t epoch, uint16_t *date, uint16_t *time)
2303 {
2304  tm_t *t = gmtime((time_t *) &epoch);
2305 
2306 /* Pack date and time into a uint32_t variable */
2307  *date = ((uint16_t)(t->tm_year - 80) << 9)
2308  | (((uint16_t)t->tm_mon+1) << 5)
2309  | (((uint16_t)t->tm_mday));
2310 
2311  *time = ((uint16_t)t->tm_hour << 11)
2312  | ((uint16_t)t->tm_min << 5)
2313  | ((uint16_t)t->tm_sec >> 1);
2314 }
2315 
2316 
2326 MEMSPACE
2328 {
2329  FILE *stream;
2330  FIL *fh;
2331 
2332  if(isatty( fileno ))
2333  {
2334  errno = EBADF;
2335  return(NULL);
2336  }
2337 
2338 // checks if fileno out of bounds
2339  stream = fileno_to_stream(fileno);
2340  if( stream == NULL )
2341  return(NULL);
2342 
2343  fh = fdev_get_udata(stream);
2344  if(fh == NULL)
2345  {
2346  errno = EBADF;
2347  return(NULL);
2348  }
2349  return(fh);
2350 }
2351 
2352 
2360 MEMSPACE
2362 {
2363  FILE *stream;
2364  FIL *fh;
2365 
2366  if(isatty( fileno ))
2367  {
2368  errno = EBADF;
2369  return(-1);
2370  }
2371 
2372 // checks if fileno out of bounds
2373  stream = fileno_to_stream(fileno);
2374  if(stream == NULL)
2375  {
2376  return(-1);
2377  }
2378 
2379  fh = fdev_get_udata(stream);
2380 
2381  if(fh != NULL)
2382  {
2383  safefree(fh);
2384  }
2385 
2386  if(stream->buf != NULL && stream->flags & __SMALLOC)
2387  {
2388  safefree(stream->buf);
2389  }
2390 
2391  __iob[fileno] = NULL;
2392  safefree(stream);
2393  return(fileno);
2394 }
2395 
2396 
2397 // =============================================
2403 MEMSPACE
2405 {
2406  int i;
2407  FILE *stream;
2408  FIL *fh;
2409 
2410  for(i=0;i<MAX_FILES;++i)
2411  {
2412  if(isatty(i))
2413  continue;
2414  if( __iob[i] == NULL)
2415  {
2416  stream = (FILE *) safecalloc(sizeof(FILE),1);
2417  if(stream == NULL)
2418  {
2419  errno = ENOMEM;
2420  return(-1);
2421  }
2422  fh = (FIL *) safecalloc(sizeof(FIL),1);
2423  if(fh == NULL)
2424  {
2425  safefree(stream);
2426  errno = ENOMEM;
2427  return(-1);
2428  }
2429 
2430  __iob[i] = stream;
2431  fdev_set_udata(stream, (void *) fh);
2432  return(i);
2433  }
2434  }
2435  errno = ENFILE;
2436  return(-1);
2437 }
2438 
2439 
2467 MEMSPACE
2468 int posix_fopen_modes_to_open(const char *mode)
2469 {
2470  int flag = 0;
2471 
2472  if(modecmp(mode,"r") || modecmp(mode,"rb"))
2473  {
2474  flag = O_RDONLY;
2475  return(flag);
2476  }
2477  if(modecmp(mode,"r+") || modecmp(mode, "r+b" ) || modecmp(mode, "rb+" ))
2478  {
2479  flag = O_RDWR | O_TRUNC;
2480  return(flag);
2481  }
2482  if(modecmp(mode,"w") || modecmp(mode,"wb"))
2483  {
2484  flag = O_WRONLY | O_CREAT | O_TRUNC;
2485  return(flag);
2486  }
2487  if(modecmp(mode,"w+") || modecmp(mode, "w+b" ) || modecmp(mode, "wb+" ))
2488  {
2489  flag = O_RDWR | O_CREAT | O_TRUNC;
2490  return(flag);
2491  }
2492  if(modecmp(mode,"a") || modecmp(mode,"ab"))
2493  {
2494  flag = O_WRONLY | O_CREAT | O_APPEND;
2495  return(flag);
2496  }
2497  if(modecmp(mode,"a+") || modecmp(mode, "a+b" ) || modecmp(mode, "ab+" ))
2498  {
2499  flag = O_RDWR | O_CREAT | O_APPEND;
2500  return(-1);
2501  }
2502  return(-1); // nvalid mode
2503 }
2504 
2505 
2506 // =============================================
2507 // =============================================
2509 // =============================================
2510 // =============================================
2511 
2516 MEMSPACE
2517 static void _fprintf_putc(struct _printf_t *p, char ch)
2518 {
2519  p->sent++;
2520  fputc(ch, (FILE *) p->buffer);
2521 }
2522 
2523 
2531 MEMSPACE
2532 int
2533 fprintf(FILE *fp, const char *format, ...)
2534 {
2535  printf_t fn;
2536  va_list va;
2537 
2538  fn.put = _fprintf_putc;
2539  fn.sent = 0;
2540  fn.buffer = (void *) fp;
2541 
2542  va_start(va, format);
2543  _printf_fn(&fn, format, va);
2544  va_end(va);
2545 
2546  return ((int)fn.sent);
2547 }
__file::buf
char * buf
Definition: posix.h:160
FATFS
Definition: ff.h:132
FR_NOT_ENOUGH_CORE
@ FR_NOT_ENOUGH_CORE
Definition: ff.h:294
FILINFO::ftime
WORD ftime
Definition: ff.h:250
fatfs.h
__SRD
#define __SRD
Definition: posix.h:163
stat::st_size
off_t st_size
Definition: posix.h:114
fopen
MEMSPACE FILE * fopen(const char *path, const char *mode)
POSIX Open a file with path name and ascii file mode string.
Definition: posix.c:801
_printf_t::buffer
void * buffer
Definition: mathio.h:83
FA_OPEN_ALWAYS
#define FA_OPEN_ALWAYS
Definition: ff.h:395
_FDEV_ERR
#define _FDEV_ERR
Definition: posix.h:300
EBUSY
@ EBUSY
Definition: posix.h:79
stat
POSIX stat structure.
Definition: posix.h:105
_fprintf_putc
static MEMSPACE void _fprintf_putc(struct _printf_t *p, char ch)
fprintf character write function
Definition: posix.c:2517
chdir
MEMSPACE int chdir(const char *pathname)
POSIX change directory.
Definition: posix.c:1512
putc
MEMSPACE int putc(int c, FILE *stream)
Put a character to a stream See fdevopen() sets stream->put get for TTY devices.
Definition: posix.c:414
stat::st_mtime
time_t st_mtime
Definition: posix.h:118
S_ISDIR
#define S_ISDIR(mode)
Definition: posix.h:219
printf
MEMSPACE int printf(const char *format,...)
MKFS_PARM
Definition: ff.h:264
FATFS_R
#define FATFS_R
FATFS open modes.
Definition: posix.h:246
errno
int errno
Note: fdevopen assigns stdin,stdout,stderr.
Definition: posix.c:129
stdin
#define stdin
define stdin, stdout and stderr
Definition: posix.h:273
modecmp
#define modecmp(str, pat)
used in posix.c to compare to ascii file modes
Definition: posix.h:242
stderr
#define stderr
Definition: posix.h:275
dump_stat
MEMSPACE void dump_stat(struct stat *sp)
Display struct stat, from POSIX stat(0 or fstat(), in ASCII. NOT POSIX.
Definition: posix.c:1283
isatty
MEMSPACE int isatty(int fileno)
Test POSIX fileno if it is a Serial Console/TTY.
Definition: posix.c:195
FR_DISK_ERR
@ FR_DISK_ERR
Definition: ff.h:278
FR_INVALID_OBJECT
@ FR_INVALID_OBJECT
Definition: ff.h:286
__SUNGET
#define __SUNGET
Definition: posix.h:169
tm::tm_min
int tm_min
Definition: time.h:43
fat_time_to_unix
MEMSPACE time_t fat_time_to_unix(uint16_t date, uint16_t time)
Convert FatFs file date and time to POSIX epoch seconds. NOT POSIX.
Definition: posix.c:2276
__file::put
int(* put)(char, struct __file *)
Definition: posix.h:180
mctime
MEMSPACE char * mctime(time_t timev)
Display Ascii formatted time from timev seconds NOT POSIX.
Definition: posix.c:1346
FIL
Definition: ff.h:205
__SERR
#define __SERR
Definition: posix.h:167
FR_EXIST
@ FR_EXIST
Definition: ff.h:285
f_write
FRESULT f_write(FIL *fp, const void *buff, UINT btw, UINT *bw)
Definition: ff.c:3990
MATCH
MEMSPACE int MATCH(char *str, char *pat)
Compare two strings.
Definition: parsing.c:143
f_tell
#define f_tell(fp)
Definition: ff.h:341
_FDEV_SETUP_READ
#define _FDEV_SETUP_READ
Definition: posix.h:302
open
MEMSPACE int open(const char *pathname, int flags)
POSIX Open a file with integer mode flags.
Definition: posix.c:916
f_stat
FRESULT f_stat(const TCHAR *path, FILINFO *fno)
Definition: ff.c:4755
stat::st_uid
uid_t st_uid
Definition: posix.h:111
_printf_t::put
void(* put)(struct _printf_t *, char)
Definition: mathio.h:82
f_rename
FRESULT f_rename(const TCHAR *path_old, const TCHAR *path_new)
Definition: ff.c:5108
sync
MEMSPACE void sync(void)
POSIX Sync all pending file changes and metadata on ALL files.
Definition: posix.c:1091
mkfs
int mkfs(char *name)
Formt SD card.
Definition: posix.c:1960
FR_NO_FILE
@ FR_NO_FILE
Definition: ff.h:281
tolower
MEMSPACE int WEAK_ATR tolower(int c)
Convert character to lower case, only if it is upper case.
Definition: stringsup.c:91
FA_WRITE
#define FA_WRITE
Definition: ff.h:391
fatfs_putc
MEMSPACE int fatfs_putc(char c, FILE *stream)
Private FatFs function called by fputc() to put a byte from file stream NOT POSIX open() assigns stre...
Definition: posix.c:2122
EPERM
@ EPERM
Definition: posix.h:64
MEMSPACE
#define MEMSPACE
Definition: user_config.h:17
write
MEMSPACE ssize_t write(int fd, const void *buf, size_t count)
POSIX Write count bytes from *buf to fileno fd.
Definition: posix.c:1202
O_CREAT
#define O_CREAT
Definition: posix.h:193
timegm
MEMSPACE time_t timegm(tm_t *t)
Convert tm_t structure as GMT time into GMT seconds since 1900. All calculactions are in GMT regardle...
Definition: time.c:348
BYTE
unsigned char BYTE
Definition: ff.h:54
f_utime
FRESULT f_utime(const TCHAR *path, const FILINFO *fno)
Definition: ff.c:5265
safecalloc
void * safecalloc(int size, int elements)
Safe Alloc - Display Error message if Calloc fails.
Definition: ram.c:122
FR_NO_PATH
@ FR_NO_PATH
Definition: ff.h:282
fgets
MEMSPACE char * fgets(char *str, int size, FILE *stream)
get a string from stdin See fdevopen() sets stream->put get for TTY devices
Definition: posix.c:432
new_file_descriptor
MEMSPACE int new_file_descriptor(void)
Allocate a POSIX FILE descriptor. NOT POSIX.
Definition: posix.c:2404
fputc
MEMSPACE int fputc(int c, FILE *stream)
Put a byte to TTY device or FatFs file stream open() or fopen() sets stream->put = fatfs_outc() for F...
Definition: posix.c:287
closedir
int closedir(DIR *dirp)
POSIX closedir.
Definition: posix.c:1748
fprintf
MEMSPACE int fprintf(FILE *fp, const char *format,...)
fprintf function Example user defined printf function using fputc for I/O This method allows I/O to d...
Definition: posix.c:2533
f_open
FRESULT f_open(FIL *fp, const TCHAR *path, BYTE mode)
Definition: ff.c:3697
clrerror
MEMSPACE void clrerror(FILE *stream)
clrerror resets stream EOF and error flags
Definition: posix.c:1820
_dp
static DIR _dp
POSIX opendir.
Definition: posix.c:1767
O_APPEND
#define O_APPEND
Definition: posix.h:199
gmtime
MEMSPACE tm_t * gmtime(time_t *tp)
Convert epoch GMT time_t *tp into POSIX static tm_t *t.
Definition: time.c:471
chmod
MEMSPACE int chmod(const char *pathname, mode_t mode)
POSIX chmod function - change file access permission Unfortunately file f_open modes and f_chmod mode...
Definition: posix.c:1538
ctime_gm
MEMSPACE char * ctime_gm(time_t *tp)
GMT version of POSIX ctime().
Definition: time.c:441
FA_CREATE_ALWAYS
#define FA_CREATE_ALWAYS
Definition: ff.h:394
stringsup.h
Various string and character functions.
ENFILE
@ ENFILE
Definition: posix.h:86
S_IWGRP
#define S_IWGRP
Definition: posix.h:231
FILINFO::fdate
WORD fdate
Definition: ff.h:249
EROFS
@ EROFS
Definition: posix.h:93
S_ISREG
#define S_ISREG(mode)
Definition: posix.h:222
f_mkfs
FRESULT f_mkfs(const TCHAR *path, const MKFS_PARM *opt, void *work, UINT len)
Definition: ff.c:5838
free_file_descriptor
MEMSPACE int free_file_descriptor(int fileno)
Free POSIX fileno FILE descriptor. NOT POSIX.
Definition: posix.c:2361
close
MEMSPACE int close(int fileno)
POSIX Close a file with fileno handel.
Definition: posix.c:701
fatfs_to_fileno
MEMSPACE int fatfs_to_fileno(FIL *fh)
Convert FatFS file handle to POSIX fileno. NOT POSIX.
Definition: posix.c:2238
tm::tm_sec
int tm_sec
Definition: time.h:42
rename
MEMSPACE int rename(const char *oldpath, const char *newpath)
POSIX rename a file by name.
Definition: posix.c:1677
fwrite
MEMSPACE size_t fwrite(const void *ptr, size_t size, size_t nmemb, FILE *stream)
POSIX write nmemb elements from buf, size bytes each, to the stream fd.
Definition: posix.c:890
safefree
void safefree(void *p)
Safe free - Only free a pointer if it is in malloc memory range.
Definition: ram.c:158
strerror
MEMSPACE char WEAK_ATR * strerror(int errnum)
POSIX strerror() - convert POSIX errno to text with user message.
Definition: posix.c:1876
stat::st_ctime
time_t st_ctime
Definition: posix.h:119
ungetc
MEMSPACE int ungetc(int c, FILE *stream)
Un-Get byte from a TTY device or FatFs file stream.
Definition: posix.c:380
_printf_t
undefine any potential macro version of these functions
Definition: mathio.h:80
_FDEV_SETUP_WRITE
#define _FDEV_SETUP_WRITE
Definition: posix.h:303
fdev_set_udata
#define fdev_set_udata(stream, u)
device IO udata
Definition: posix.h:295
sys_errlist
const char * sys_errlist[]
POSIX error messages for each errno value.
Definition: posix.c:143
tm::tm_mon
int tm_mon
Definition: time.h:46
FR_WRITE_PROTECTED
@ FR_WRITE_PROTECTED
Definition: ff.h:287
FA_OPEN_EXISTING
#define FA_OPEN_EXISTING
Definition: ff.h:392
tm::tm_mday
int tm_mday
Definition: time.h:45
S_IWUSR
#define S_IWUSR
Definition: posix.h:226
rmdir
MEMSPACE int rmdir(const char *pathname)
POSIX delete a directory.
Definition: posix.c:1701
f_unlink
FRESULT f_unlink(const TCHAR *path)
Definition: ff.c:4930
unix_time_to_fat
MEMSPACE void unix_time_to_fat(time_t epoch, uint16_t *date, uint16_t *time)
Convert Linux POSIX time_t to FAT32 date and time. NOT POSIX.
Definition: posix.c:2302
UINT
unsigned int UINT
Definition: ff.h:53
tm::tm_hour
int tm_hour
Definition: time.h:44
__file::get
int(* get)(struct __file *)
Definition: posix.h:181
__SEOF
#define __SEOF
Definition: posix.h:168
stat
MEMSPACE int stat(char *name, struct stat *buf)
POSIX stat - get file status of named file.
Definition: posix.c:1368
baseext
MEMSPACE char * baseext(char *str)
File extention of a file name. NOT POSIX.
Definition: posix.c:1490
ENXIO
@ ENXIO
Definition: posix.h:69
getchar
MEMSPACE int getchar()
functions normally defined as macros
Definition: posix.c:347
S_IFREG
#define S_IFREG
Definition: posix.h:209
_de
static dirent_t _de
POSIX opendir.
Definition: posix.c:1787
readdir
dirent_t * readdir(DIR *dirp)
Definition: posix.c:1788
safemalloc
void * safemalloc(size_t size)
Safe Malloc - Display Error message if Malloc fails.
Definition: ram.c:139
ssize_t
int32_t ssize_t
Definition: posix.h:55
_printf_t::sent
int sent
Definition: mathio.h:85
SEEK_CUR
#define SEEK_CUR
Definition: posix.h:256
dirent::d_name
char d_name[MAX_NAME_LEN]
Definition: posix.h:146
FATFS_W
#define FATFS_W
Definition: posix.h:247
__SSTR
#define __SSTR
Definition: posix.h:165
NULL
#define NULL
Definition: user_config.h:85
FR_LOCKED
@ FR_LOCKED
Definition: ff.h:293
mode_t
uint32_t mode_t
Definition: posix.h:47
FR_NOT_ENABLED
@ FR_NOT_ENABLED
Definition: ff.h:289
O_TRUNC
#define O_TRUNC
Definition: posix.h:198
__file::size
int size
Definition: posix.h:178
posix_fopen_modes_to_open
MEMSPACE int posix_fopen_modes_to_open(const char *mode)
Convert POSIX fopen mode to POSIX open mode flags. NOT POSIX.
Definition: posix.c:2468
fatfs_getc
MEMSPACE int fatfs_getc(FILE *stream)
Private FatFs function called by fgetc() to get a byte from file stream FIXME buffer this function ca...
Definition: posix.c:2031
fputs
MEMSPACE int fputs(const char *str, FILE *stream)
put a string to stdout See fdevopen() sets stream->put get for TTY devices
Definition: posix.c:470
fileno_to_stream
MEMSPACE FILE * fileno_to_stream(int fileno)
Convert POSIX fileno to POSIX FILE stream pointer. NOT POSIX.
Definition: posix.c:772
__file
FILE type structure.
Definition: posix.h:158
strlen
MEMSPACE size_t WEAK_ATR strlen(const char *str)
String Length.
Definition: stringsup.c:144
FA_READ
#define FA_READ
Definition: ff.h:390
O_RDWR
#define O_RDWR
Definition: posix.h:192
dirent
Definition: posix.h:136
syncfs
MEMSPACE int syncfs(int fd)
POSIX Sync pending file changes and metadata for specified fileno.
Definition: posix.c:1119
basename
MEMSPACE char * basename(char *str)
POSIX Basename of filename.
Definition: posix.c:1469
f_getcwd
FRESULT f_getcwd(TCHAR *buff, UINT len)
Definition: ff.c:4308
perror
MEMSPACE void perror(const char *s)
POSIX perror() - convert POSIX errno to text with user message.
Definition: posix.c:1850
f_size
#define f_size(fp)
Definition: ff.h:342
FATFS_X
#define FATFS_X
Definition: posix.h:248
read
MEMSPACE ssize_t read(int fd, const void *buf, size_t count)
POSIX read count bytes from *buf to fileno fd.
Definition: posix.c:1028
f_opendir
FRESULT f_opendir(DIR *dp, const TCHAR *path)
Definition: ff.c:4571
ftruncate
MEMSPACE int ftruncate(int fd, off_t length)
POSIX truncate open file to length.
Definition: posix.c:848
__iob
FILE * __iob[MAX_FILES]
POSIX fileno to POSIX FILE stream table.
Definition: posix.c:138
opendir
DIR * opendir(const char *pathdir)
Definition: posix.c:1768
FR_DENIED
@ FR_DENIED
Definition: ff.h:284
stdout
#define stdout
Definition: posix.h:274
fclose
MEMSPACE int fclose(FILE *stream)
POSIX close a file stream.
Definition: posix.c:1261
ftell
MEMSPACE long ftell(FILE *stream)
POSIX file position of open stream.
Definition: posix.c:600
FR_MKFS_ABORTED
@ FR_MKFS_ABORTED
Definition: ff.h:291
SEEK_END
#define SEEK_END
Definition: posix.h:257
fdev_get_udata
#define fdev_get_udata(stream)
Definition: posix.h:296
__SWR
#define __SWR
Definition: posix.h:164
FR_NOT_READY
@ FR_NOT_READY
Definition: ff.h:280
f_lseek
FRESULT f_lseek(FIL *fp, FSIZE_t ofs)
Definition: ff.c:4408
stat::st_mode
mode_t st_mode
Definition: posix.h:109
__file::flags
uint8_t flags
Definition: posix.h:162
__file::unget
unsigned char unget
Definition: posix.h:161
EINVAL
@ EINVAL
Definition: posix.h:85
FILINFO::fattrib
BYTE fattrib
Definition: ff.h:251
mkdir
MEMSPACE int mkdir(const char *pathname, mode_t mode)
POSIX make a directory.
Definition: posix.c:1647
ferror
MEMSPACE int ferror(FILE *stream)
ferror reports if the stream has an error flag set
Definition: posix.c:1833
fseek
MEMSPACE int fseek(FILE *stream, long offset, int whence)
POSIX seek to file possition.
Definition: posix.c:558
off_t
uint32_t off_t
Definition: posix.h:51
time
MEMSPACE time_t time(time_t *t)
Return second from epoch - POSIX function.
Definition: time.c:843
FR_NO_FILESYSTEM
@ FR_NO_FILESYSTEM
Definition: ff.h:290
O_WRONLY
#define O_WRONLY
Definition: posix.h:191
AM_SYS
#define AM_SYS
Definition: ff.h:417
EIO
@ EIO
Definition: posix.h:68
putchar
MEMSPACE int putchar(int c)
put a character to stdout See fdevopen() sets stream->put get for TTY devices
Definition: posix.c:363
f_chdir
FRESULT f_chdir(const TCHAR *path)
Definition: ff.c:4246
SEEK_SET
#define SEEK_SET
Seek offset macros.
Definition: posix.h:255
rewind
MEMSPACE void rewind(FILE *stream)
POSIX rewind file to the beginning.
Definition: posix.c:681
S_IWOTH
#define S_IWOTH
Definition: posix.h:236
truncate
MEMSPACE int truncate(const char *path, off_t length)
POSIX truncate named file to length.
Definition: posix.c:1164
FR_INT_ERR
@ FR_INT_ERR
Definition: ff.h:279
FRESULT
FRESULT
Definition: ff.h:276
fatfs_to_errno
MEMSPACE int fatfs_to_errno(FRESULT Result)
Convert FafFs error result to POSIX errno. NOT POSIX.
Definition: posix.c:2163
ENOSPC
@ ENOSPC
Definition: posix.h:91
DIR
Definition: ff.h:228
getcwd
MEMSPACE char * getcwd(char *pathname, int len)
POSIX get current working directory.
Definition: posix.c:1623
AM_DIR
#define AM_DIR
Definition: ff.h:418
fgetpos
MEMSPACE int fgetpos(FILE *stream, size_t *pos)
POSIX get position of file stream.
Definition: posix.c:534
EACCES
@ EACCES
Definition: posix.h:76
mmc_init
MEMSPACE int mmc_init(int verbose)
Initialize MMC and FatFs interface, display diagnostics.
Definition: mmc_hal.c:219
stat::st_atime
time_t st_atime
Definition: posix.h:117
ENOENT
@ ENOENT
Definition: posix.h:65
FR_OK
@ FR_OK
Definition: ff.h:277
FIL::fptr
FSIZE_t fptr
Definition: ff.h:209
utimbuf::modtime
time_t modtime
Definition: posix.h:127
strerror_r
MEMSPACE char * strerror_r(int errnum, char *buf, size_t buflen)
POSIX strerror_r() - convert POSIX errno to text with user message.
Definition: posix.c:1893
fileno
MEMSPACE int fileno(FILE *stream)
Convert POSIX stream pointer to POSIX fileno (index of __iob[])
Definition: posix.c:741
O_ACCMODE
#define O_ACCMODE
POSIX open modes - no other combination are allowed.
Definition: posix.h:189
ENOMEM
@ ENOMEM
Definition: posix.h:75
FILINFO::fname
TCHAR fname[FF_LFN_BUF+1]
Definition: ff.h:254
f_mount
FRESULT f_mount(FATFS *fs, const TCHAR *path, BYTE opt)
Definition: ff.c:3649
FR_INVALID_PARAMETER
@ FR_INVALID_PARAMETER
Definition: ff.h:296
S_IFDIR
#define S_IFDIR
Definition: posix.h:206
FR_TOO_MANY_OPEN_FILES
@ FR_TOO_MANY_OPEN_FILES
Definition: ff.h:295
lseek
MEMSPACE off_t lseek(int fileno, off_t position, int whence)
POSIX seek to file position.
Definition: posix.c:633
tm
POSIX struct tm.
Definition: time.h:40
f_truncate
FRESULT f_truncate(FIL *fp)
Definition: ff.c:4880
tm::tm_year
int tm_year
Definition: time.h:47
FILINFO::fsize
FSIZE_t fsize
Definition: ff.h:248
fdevopen
MEMSPACE FILE * fdevopen(int(*put)(char, FILE *), int(*get)(FILE *))
Device open functions.
Definition: posix.c:1912
fileno_to_fatfs
MEMSPACE FIL * fileno_to_fatfs(int fileno)
Convert POSIX fileno to FatFS handle NOT POSIX.
Definition: posix.c:2327
time_t
uint32_t time_t
type of EPOCH result.
Definition: time.h:34
unlink
MEMSPACE int unlink(const char *pathname)
POSIX delete a file.
Definition: posix.c:1723
FR_INVALID_NAME
@ FR_INVALID_NAME
Definition: ff.h:283
MAX_FILES
#define MAX_FILES
Maximum number of POSIX file handles.
Definition: posix.h:264
strncpy
MEMSPACE WEAK_ATR char * strncpy(char *dest, const char *src, size_t size)
copy a string of at most N characters
Definition: stringsup.c:179
_FDEV_SETUP_RW
#define _FDEV_SETUP_RW
Definition: posix.h:304
stat::st_gid
gid_t st_gid
Definition: posix.h:112
EBADF
@ EBADF
Definition: posix.h:72
f_closedir
FRESULT f_closedir(DIR *dp)
Definition: ff.c:4637
feof
MEMSPACE int feof(FILE *stream)
feof reports if the stream is at EOF
Definition: posix.c:516
EOF
#define EOF
Definition: ff.h:349
utime
MEMSPACE int utime(const char *filename, const struct utimbuf *times)
Set Modification and Access time of a file.
Definition: posix.c:1432
fread
MEMSPACE size_t fread(void *ptr, size_t size, size_t nmemb, FILE *stream)
POSIX read nmemb elements from buf, size bytes each, to the stream fd.
Definition: posix.c:823
FILINFO
Definition: ff.h:247
RES_OK
@ RES_OK
Definition: diskio.h:31
dirname
MEMSPACE int dirname(char *str)
POSIX directory name of a filename. Return the index of the last '/' character.
Definition: posix.c:1577
f_chmod
FRESULT f_chmod(const TCHAR *path, BYTE attr, BYTE mask)
Definition: ff.c:5218
posix.h
POSIX wrapper for FatFS.
FM_FAT32
#define FM_FAT32
Definition: ff.h:403
O_RDONLY
#define O_RDONLY
Definition: posix.h:190
utimbuf
POSIX utimbuf structure.
Definition: posix.h:124
__file::len
int len
Definition: posix.h:179
AM_RDO
#define AM_RDO
Definition: ff.h:415
fgetc
MEMSPACE int fgetc(FILE *stream)
Get byte from a TTY device or FatFs file stream open() or fopen() sets stream->get = fatfs_getc() for...
Definition: posix.c:216
FR_TIMEOUT
@ FR_TIMEOUT
Definition: ff.h:292
EBADMSG
@ EBADMSG
Definition: posix.h:98
FR_INVALID_DRIVE
@ FR_INVALID_DRIVE
Definition: ff.h:288
EMFILE
@ EMFILE
Definition: posix.h:87
puts
MEMSPACE int puts(const char *str)
put a string to stdout See fdevopen() sets stream->put get for TTY devices
Definition: posix.c:493
f_mkdir
FRESULT f_mkdir(const TCHAR *path)
Definition: ff.c:5024
f_readdir
FRESULT f_readdir(DIR *dp, FILINFO *fno)
Definition: ff.c:4667
f_read
FRESULT f_read(FIL *fp, void *buff, UINT btr, UINT *br)
Definition: ff.c:3889
f_sync
FRESULT f_sync(FIL *fp)
Definition: ff.c:4112
f_close
FRESULT f_close(FIL *fp)
Definition: ff.c:4193
_printf_fn
MEMSPACE void _printf_fn(printf_t *fn, __memx const char *fmt, va_list va)
vsnprintf function
Definition: printf.c:806
WEAK_ATR
#define WEAK_ATR
Definition: stringsup.h:33
fsetpos
MEMSPACE int fsetpos(FILE *stream, size_t *pos)
POSIX set position of file stream.
Definition: posix.c:585
__SMALLOC
#define __SMALLOC
Definition: posix.h:170