4
0
mirror of git://sourceware.org/git/newlib-cygwin.git synced 2025-02-13 12:39:22 +08:00
Vladimir Isaev 6d5331054e arc: libgloss: Introduce hostlink interface
There is a special interface built in ARC simulators (such as
nSIM) called MetaWare hostlink IO which can be used to implement
system calls. This commit adds support for this interface to the
ARC port of libgloss.

Here is an example of using this interface:

    $ arc-elf32-gcc -mcpu=hs -specs=hl.specs main.c -o main
    $ nsimdrv -tcf $NSIM_HOME/etc/tcf/templates/hs48_full.tcf main
    Hello, World!

Signed-off-by: Vladimir Isaev <vvisaev@gmail.com>
2024-05-22 14:25:44 -04:00

90 lines
2.1 KiB
C

/*
* hl_open.c -- provide _open().
*
* Copyright (c) 2024 Synopsys Inc.
*
* The authors hereby grant permission to use, copy, modify, distribute,
* and license this software and its documentation for any purpose, provided
* that existing copyright notices are retained in all copies and that this
* notice is included verbatim in any distributions. No written agreement,
* license, or royalty fee is required for any of the authorized uses.
* Modifications to this software may be copyrighted by their authors
* and need not follow the licensing terms described here, provided that
* the new terms are clearly indicated on the first page of each file where
* they apply.
*
*/
#include <errno.h>
#include <stdarg.h>
#include <stdint.h>
#include <sys/stat.h>
#include <sys/types.h>
#include <fcntl.h>
#include "hl_toolchain.h"
#include "hl_api.h"
/* Map newlib open flags into Hostlink IO ones. */
static __always_inline uint32_t
_hl_open_flags_map (int flags)
{
uint32_t hl_flags = 0;
hl_flags |= (flags & O_RDONLY) ? 0x0000 : 0;
hl_flags |= (flags & O_WRONLY) ? 0x0001 : 0;
hl_flags |= (flags & O_RDWR) ? 0x0002 : 0;
hl_flags |= (flags & O_APPEND) ? 0x0008 : 0;
hl_flags |= (flags & O_CREAT) ? 0x0100 : 0;
hl_flags |= (flags & O_TRUNC) ? 0x0200 : 0;
hl_flags |= (flags & O_EXCL) ? 0x0400 : 0;
return hl_flags;
}
/* Open file on host. Implements HL_SYSCALL_OPEN. */
static __always_inline int
_hl_open (const char *path, int flags, mode_t mode)
{
int32_t fd;
uint32_t host_errno;
uint32_t hl_flags = _hl_open_flags_map (flags);
volatile __uncached char *p;
p = _hl_message (HL_SYSCALL_OPEN, "sii:ii",
path, /* s */
(uint32_t) hl_flags, /* i */
(uint32_t) mode, /* i */
(uint32_t *) &fd, /* :i */
(uint32_t *) &host_errno /* :i */);
if (p == NULL)
{
errno = ETIMEDOUT;
fd = -1;
}
else if (fd < 0)
{
errno = host_errno;
fd = -1;
}
_hl_delete ();
return fd;
}
int
_open (const char *path, int flags, ...)
{
va_list ap;
mode_t mode = 0;
va_start (ap, flags);
if (flags & O_CREAT)
mode = va_arg (ap, mode_t);
return _hl_open (path, flags, mode);
}