mirror of
git://sourceware.org/git/newlib-cygwin.git
synced 2025-02-13 04:29:09 +08:00
106 lines
2.1 KiB
C
106 lines
2.1 KiB
C
|
/*
|
||
|
* hl_read.c -- provide _read().
|
||
|
*
|
||
|
* 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 <stdint.h>
|
||
|
#include <stddef.h>
|
||
|
#include <unistd.h>
|
||
|
|
||
|
#include "hl_toolchain.h"
|
||
|
#include "hl_api.h"
|
||
|
|
||
|
|
||
|
/* Read one chunk. Implements HL_SYSCALL_READ. */
|
||
|
static ssize_t
|
||
|
_hl_read (int fd, void *buf, size_t count)
|
||
|
{
|
||
|
ssize_t ret;
|
||
|
int32_t hl_n;
|
||
|
uint32_t host_errno;
|
||
|
volatile __uncached char *p;
|
||
|
|
||
|
p = _hl_message (HL_SYSCALL_READ, "ii:i",
|
||
|
(uint32_t) fd, /* i */
|
||
|
(uint32_t) count, /* i */
|
||
|
(uint32_t *) &hl_n /* :i */);
|
||
|
|
||
|
if (p == NULL)
|
||
|
{
|
||
|
errno = ETIMEDOUT;
|
||
|
ret = -1;
|
||
|
}
|
||
|
else if (hl_n < 0)
|
||
|
{
|
||
|
p = _hl_unpack_int (p, &host_errno);
|
||
|
errno = p == NULL ? EIO : host_errno;
|
||
|
ret = -1;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
uint32_t n;
|
||
|
|
||
|
p = _hl_unpack_ptr (p, buf, &n);
|
||
|
ret = n;
|
||
|
|
||
|
if (p == NULL || n != (uint32_t) hl_n)
|
||
|
{
|
||
|
errno = EIO;
|
||
|
ret = -1;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
_hl_delete ();
|
||
|
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
ssize_t
|
||
|
_read (int fd, void *buf, size_t count)
|
||
|
{
|
||
|
const uint32_t hl_iochunk = _hl_iochunk_size ();
|
||
|
size_t to_read = count;
|
||
|
size_t offset = 0;
|
||
|
ssize_t ret = 0;
|
||
|
|
||
|
while (to_read > hl_iochunk)
|
||
|
{
|
||
|
ret = _hl_read (fd, (char *) buf + offset, hl_iochunk);
|
||
|
|
||
|
if (ret < 0)
|
||
|
return ret;
|
||
|
|
||
|
offset += ret;
|
||
|
|
||
|
if (ret != (ssize_t) hl_iochunk)
|
||
|
return offset;
|
||
|
|
||
|
to_read -= hl_iochunk;
|
||
|
}
|
||
|
|
||
|
if (to_read)
|
||
|
{
|
||
|
ret = _hl_read (fd, (char *) buf + offset, to_read);
|
||
|
|
||
|
if (ret < 0)
|
||
|
return ret;
|
||
|
|
||
|
ret += offset;
|
||
|
}
|
||
|
|
||
|
return ret;
|
||
|
}
|