351 lines
8.0 KiB
C
Raw Normal View History

/*
2021-03-08 18:19:04 +08:00
* Copyright (c) 2006-2021, RT-Thread Development Team
*
2018-10-14 19:28:18 +08:00
* SPDX-License-Identifier: Apache-2.0
*
* Change Logs:
* Date Author Notes
* 2012-11-23 Yihui The first version
2013-11-24 23:11:17 +08:00
* 2013-11-24 aozima fixed _sys_read()/_sys_write() issues.
2015-01-28 14:14:30 +08:00
* 2014-08-03 bernard If using msh, use system() implementation
* in msh.
2021-03-08 18:19:04 +08:00
* 2020-08-05 Meco Man fixed _sys_flen() compiling-warning when
* RT_USING_DFS is not defined
2021-02-13 13:30:40 +08:00
* 2020-02-13 Meco Man re-implement exit() and abort()
2021-02-14 18:46:55 +08:00
* 2020-02-14 Meco Man implement _sys_tmpnam()
*/
#include <rt_sys.h>
2021-02-14 18:47:48 +08:00
#include <rtthread.h>
#include <string.h>
2021-10-26 00:51:32 -04:00
#include <fcntl.h>
#include <unistd.h>
#include <sys/stat.h>
#ifdef RT_USING_POSIX_DEVIO
#include "libc.h"
#endif /* RT_USING_POSIX_DEVIO */
#define DBG_TAG "armlibc.syscalls"
#define DBG_LVL DBG_INFO
#include <rtdbg.h>
2018-09-23 12:08:44 +08:00
#ifdef __CLANG_ARM
__asm(".global __use_no_semihosting\n\t");
#else
#pragma import(__use_no_semihosting_swi)
2021-11-15 11:22:49 -05:00
#endif /* __CLANG_ARM */
/* Standard IO device handles. */
#define STDIN 0
#define STDOUT 1
#define STDERR 2
/* Standard IO device name defines. */
const char __stdin_name[] = "STDIN";
const char __stdout_name[] = "STDOUT";
const char __stderr_name[] = "STDERR";
/**
* required by fopen() and freopen().
*
* @param name - file name with path.
* @param openmode - a bitmap hose bits mostly correspond directly to
* the ISO mode specification.
* @return -1 if an error occurs.
*/
FILEHANDLE _sys_open(const char *name, int openmode)
{
2021-11-15 11:22:49 -05:00
#ifdef DFS_USING_POSIX
2013-11-24 23:11:17 +08:00
int fd;
int mode = O_RDONLY;
2021-11-15 11:22:49 -05:00
#endif /* DFS_USING_POSIX */
2015-01-28 14:14:30 +08:00
/* Register standard Input Output devices. */
if (strcmp(name, __stdin_name) == 0)
return (STDIN);
if (strcmp(name, __stdout_name) == 0)
return (STDOUT);
if (strcmp(name, __stderr_name) == 0)
return (STDERR);
2021-11-15 11:22:49 -05:00
#ifndef DFS_USING_POSIX
2021-10-26 00:51:32 -04:00
return 0; /* error */
#else
2015-01-28 14:14:30 +08:00
/* Correct openmode from fopen to open */
if (openmode & OPEN_PLUS)
{
if (openmode & OPEN_W)
{
mode |= (O_RDWR | O_TRUNC | O_CREAT);
}
else if (openmode & OPEN_A)
{
mode |= (O_RDWR | O_APPEND | O_CREAT);
}
else
mode |= O_RDWR;
}
else
{
if (openmode & OPEN_W)
{
mode |= (O_WRONLY | O_TRUNC | O_CREAT);
}
else if (openmode & OPEN_A)
{
mode |= (O_WRONLY | O_APPEND | O_CREAT);
2015-01-28 14:14:30 +08:00
}
}
fd = open(name, mode, 0);
2015-01-28 14:14:30 +08:00
if (fd < 0)
2021-10-26 00:51:32 -04:00
return 0; /* error */
2013-11-24 23:11:17 +08:00
else
return fd;
2021-11-15 11:22:49 -05:00
#endif /* DFS_USING_POSIX */
}
int _sys_close(FILEHANDLE fh)
{
2021-11-15 11:22:49 -05:00
#ifdef DFS_USING_POSIX
2021-10-26 00:51:32 -04:00
if (fh <= STDERR)
return 0; /* error */
2013-11-24 23:11:17 +08:00
return close(fh);
2021-10-26 00:51:32 -04:00
#else
return 0;
2021-11-15 11:22:49 -05:00
#endif /* DFS_USING_POSIX */
}
/*
* Read from a file. Can return:
* - zero if the read was completely successful
* - the number of bytes _not_ read, if the read was partially successful
* - the number of bytes not read, plus the top bit set (0x80000000), if
* the read was partially successful due to end of file
* - -1 if some error other than EOF occurred
*
* It is also legal to signal EOF by returning no data but
* signalling no error (i.e. the top-bit-set mechanism need never
* be used).
*
* So if (for example) the user is trying to read 8 bytes at a time
* from a file in which only 5 remain, this routine can do three
* equally valid things:
*
* - it can return 0x80000003 (3 bytes not read due to EOF)
* - OR it can return 3 (3 bytes not read), and then return
* 0x80000008 (8 bytes not read due to EOF) on the next attempt
* - OR it can return 3 (3 bytes not read), and then return
* 8 (8 bytes not read, meaning 0 read, meaning EOF) on the next
* attempt
*
* `mode' exists for historical reasons and must be ignored.
*/
int _sys_read(FILEHANDLE fh, unsigned char *buf, unsigned len, int mode)
{
2021-11-15 11:22:49 -05:00
#ifdef DFS_USING_POSIX
2013-11-24 23:11:17 +08:00
int size;
2015-01-28 14:14:30 +08:00
if (fh == STDIN)
{
#ifdef RT_USING_POSIX_DEVIO
if (libc_stdio_get_console() < 0)
{
LOG_W("Do not invoke standard output before initializing libc");
2021-11-12 16:47:32 -05:00
return 0; /* error, but keep going */
}
size = read(STDIN_FILENO, buf, len);
2021-11-12 16:47:32 -05:00
return 0; /* success */
#else
return 0; /* error */
#endif /* RT_USING_POSIX_DEVIO */
}
2021-11-12 16:47:32 -05:00
else if (fh == STDOUT || fh == STDERR)
{
2021-10-26 00:51:32 -04:00
return 0; /* error */
}
2013-11-24 23:11:17 +08:00
else
2021-11-12 16:47:32 -05:00
{
size = read(fh, buf, len);
if (size >= 0)
return len - size; /* success */
else
return 0; /* error */
}
2021-10-26 00:51:32 -04:00
#else
return 0; /* error */
2021-11-15 11:22:49 -05:00
#endif /* DFS_USING_POSIX */
}
/*
* Write to a file. Returns 0 on success, negative on error, and
* the number of characters _not_ written on partial success.
* `mode' exists for historical reasons and must be ignored.
2021-11-12 16:47:32 -05:00
* The return value is either:
* A positive number representing the number of characters not written
* (so any nonzero return value denotes a failure of some sort).
* A negative number indicating an error.
*/
int _sys_write(FILEHANDLE fh, const unsigned char *buf, unsigned len, int mode)
{
2021-11-15 11:22:49 -05:00
#ifdef DFS_USING_POSIX
2013-11-24 23:11:17 +08:00
int size;
2021-11-15 11:22:49 -05:00
#endif /* DFS_USING_POSIX */
2015-01-28 14:14:30 +08:00
2021-11-12 16:47:32 -05:00
if (fh == STDOUT || fh == STDERR)
{
2021-11-12 16:47:32 -05:00
#ifdef RT_USING_CONSOLE
rt_device_t console;
console = rt_console_get_device();
if (console)
{
2021-11-12 16:47:32 -05:00
rt_device_write(console, -1, buf, len);
}
2021-11-12 16:47:32 -05:00
return 0; /* success */
#else
2021-10-26 00:51:32 -04:00
return 0; /* error */
2021-11-12 16:47:32 -05:00
#endif /* RT_USING_CONSOLE */
}
else if (fh == STDIN)
{
2021-10-26 00:51:32 -04:00
return 0; /* error */
}
2013-11-24 23:11:17 +08:00
else
2021-11-12 16:47:32 -05:00
{
2021-11-15 11:22:49 -05:00
#ifdef DFS_USING_POSIX
2021-11-12 16:47:32 -05:00
size = write(fh, buf, len);
if (size >= 0)
return 0; /* success */
else
return 0; /* error */
2021-10-26 00:51:32 -04:00
#else
2021-11-12 16:47:32 -05:00
return 0; /* error */
2021-11-15 11:22:49 -05:00
#endif /* DFS_USING_POSIX */
2021-11-12 16:47:32 -05:00
}
}
/*
* Move the file position to a given offset from the file start.
* Returns >=0 on success, <0 on failure.
*/
int _sys_seek(FILEHANDLE fh, long pos)
{
2021-11-15 11:22:49 -05:00
#ifdef DFS_USING_POSIX
2013-11-24 23:11:17 +08:00
if (fh < STDERR)
2021-10-26 00:51:32 -04:00
return 0; /* error */
2013-11-24 23:11:17 +08:00
/* position is relative to the start of file fh */
return lseek(fh, pos, 0);
2021-10-26 00:51:32 -04:00
#else
return 0; /* error */
2021-11-15 11:22:49 -05:00
#endif /* DFS_USING_POSIX */
}
/**
* used by tmpnam() or tmpfile()
*/
int _sys_tmpnam(char *name, int fileno, unsigned maxlength)
{
2021-02-14 18:46:55 +08:00
rt_snprintf(name, maxlength, "tem%03d", fileno);
return 1;
}
char *_sys_command_string(char *cmd, int len)
{
/* no support */
return RT_NULL;
}
/* This function writes a character to the console. */
void _ttywrch(int ch)
{
#ifdef RT_USING_CONSOLE
char c;
c = (char)ch;
rt_kprintf(&c);
2021-10-26 00:51:32 -04:00
#endif /* RT_USING_CONSOLE */
}
2021-02-20 11:31:58 +08:00
/* for exit() and abort() */
RT_WEAK void _sys_exit(int return_code)
{
2021-02-17 23:50:12 +08:00
extern void __rt_libc_exit(int status);
__rt_libc_exit(return_code);
2021-02-17 00:18:49 +08:00
while(1);
}
/**
2013-11-24 23:11:17 +08:00
* return current length of file.
*
* @param fh - file handle
* @return file length, or -1 on failed
*/
long _sys_flen(FILEHANDLE fh)
{
2021-11-15 11:22:49 -05:00
#ifdef DFS_USING_POSIX
2019-01-24 09:05:41 +08:00
struct stat stat;
2020-08-05 09:28:10 +08:00
2019-01-24 09:05:41 +08:00
if (fh < STDERR)
2021-10-26 00:51:32 -04:00
return 0; /* error */
2019-01-24 09:05:41 +08:00
fstat(fh, &stat);
return stat.st_size;
2020-08-05 09:28:10 +08:00
#else
2021-10-26 00:51:32 -04:00
return 0;
2021-11-15 11:22:49 -05:00
#endif /* DFS_USING_POSIX */
}
int _sys_istty(FILEHANDLE fh)
{
if((STDIN <= fh) && (fh <= STDERR))
return 1;
else
return 0;
}
int remove(const char *filename)
{
2021-11-15 11:22:49 -05:00
#ifdef DFS_USING_POSIX
return unlink(filename);
2021-10-26 00:51:32 -04:00
#else
return 0; /* error */
2021-11-15 11:22:49 -05:00
#endif /* DFS_USING_POSIX */
}
#ifdef __MICROLIB
#include <stdio.h>
2021-03-08 18:19:04 +08:00
int fputc(int c, FILE *f)
{
2021-10-26 00:51:32 -04:00
#ifdef RT_USING_CONSOLE
char ch[2] = {0};
ch[0] = c;
rt_kprintf(&ch[0]);
return 1;
2021-10-26 00:51:32 -04:00
#else
return 0; /* error */
#endif /* RT_USING_CONSOLE */
}
2021-03-08 18:19:04 +08:00
int fgetc(FILE *f)
{
#ifdef RT_USING_POSIX_DEVIO
char ch;
if (libc_stdio_get_console() < 0)
{
LOG_W("Do not invoke standard output before initializing libc");
2021-10-26 00:51:32 -04:00
return 0;
}
if(read(STDIN_FILENO, &ch, 1) == 1)
return ch;
#endif /* RT_USING_POSIX_DEVIO */
2021-10-26 00:51:32 -04:00
return 0; /* error */
}
2021-10-26 00:51:32 -04:00
#endif /* __MICROLIB */