4
0
mirror of git://sourceware.org/git/newlib-cygwin.git synced 2025-01-19 12:59:21 +08:00
newlib-cygwin/libgloss/riscv/semihost-sys_fdtable.c
Venkata Ramanaiah Nalamothu c419bbb341 RISC-V: fix setting up std streams in init_semihosting()
Currently init_semihosting() assumes the return value from _open()
call as the file descriptor handle and that is incorrect.

The semihost _open() call returns the fdtable index returned by the
__add_fdentry() for the file opened.
2024-01-10 10:46:56 +01:00

87 lines
1.9 KiB
C

/*
* Copyright (C) 2020 Embecosm Limited
* SPDX-License-Identifier: BSD-2-Clause
*/
#include "semihost_fdtable.h"
#include <errno.h>
#include <fcntl.h>
#include <unistd.h>
#ifndef RISCV_MAX_OPEN_FILES
#define RISCV_MAX_OPEN_FILES 16
#endif
extern int errno;
extern int _open (const char *, int, ...);
/* fdtable keeps track of the position of each file and is used to map stdin,
stdout and stderr to STDIN_FILENO, STDOUT_FILENO and STDERR_FILENO. */
static struct fdentry fdtable[RISCV_MAX_OPEN_FILES];
/* Initialize fdtable. A handle of -1 denotes an empty entry. */
void __attribute__ ((constructor))
init_semihosting ()
{
int i;
for (i=0; i<RISCV_MAX_OPEN_FILES; i++)
fdtable[i].handle = -1;
/* Set up std streams. Note that the semihost _open() call returns an index
into the fdtable. */
/* stdin. */
i = _open (":tt", O_RDONLY);
fdtable[STDIN_FILENO].handle = fdtable[i].handle;
fdtable[STDIN_FILENO].pos = 0;
/* stdout. */
i = _open (":tt", O_WRONLY|O_CREAT|O_TRUNC);
fdtable[STDOUT_FILENO].handle = fdtable[i].handle;
fdtable[STDOUT_FILENO].pos = 0;
/* stderr. */
i = _open (":tt", O_WRONLY|O_CREAT|O_APPEND);
fdtable[STDERR_FILENO].handle = fdtable[i].handle;
fdtable[STDERR_FILENO].pos = 0;
}
/* Add entry to fdtable. */
int
__add_fdentry (int handle)
{
for (int i=0; i<RISCV_MAX_OPEN_FILES; i++)
if (fdtable[i].handle == -1)
{
fdtable[i].handle = handle;
fdtable[i].pos = 0;
return i;
}
/* Too many open files. */
errno = ENFILE;
return -1;
}
/* Return the fdentry for file or NULL if not found. */
struct fdentry *
__get_fdentry (int file)
{
if (file<0 || file>=RISCV_MAX_OPEN_FILES || fdtable[file].handle == -1)
{
errno = EBADF;
return NULL;
}
return &fdtable[file];
}
/* Remove entry from fdtable. */
void
__remove_fdentry (int file)
{
fdtable[file].handle = -1;
}