forked from mirror/qemu
You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
361 lines
9.5 KiB
C
361 lines
9.5 KiB
C
/*
|
|
* Unified Hosting Interface syscalls.
|
|
*
|
|
* Copyright (c) 2015 Imagination Technologies
|
|
*
|
|
* This library is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
* License as published by the Free Software Foundation; either
|
|
* version 2.1 of the License, or (at your option) any later version.
|
|
*
|
|
* This library is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
* Lesser General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
* License along with this library; if not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include "qemu/osdep.h"
|
|
#include "cpu.h"
|
|
#include "qemu/log.h"
|
|
#include "exec/gdbstub.h"
|
|
#include "semihosting/softmmu-uaccess.h"
|
|
#include "semihosting/semihost.h"
|
|
#include "semihosting/console.h"
|
|
#include "semihosting/syscalls.h"
|
|
#include "internal.h"
|
|
|
|
typedef enum UHIOp {
|
|
UHI_exit = 1,
|
|
UHI_open = 2,
|
|
UHI_close = 3,
|
|
UHI_read = 4,
|
|
UHI_write = 5,
|
|
UHI_lseek = 6,
|
|
UHI_unlink = 7,
|
|
UHI_fstat = 8,
|
|
UHI_argc = 9,
|
|
UHI_argnlen = 10,
|
|
UHI_argn = 11,
|
|
UHI_plog = 13,
|
|
UHI_assert = 14,
|
|
UHI_pread = 19,
|
|
UHI_pwrite = 20,
|
|
UHI_link = 22
|
|
} UHIOp;
|
|
|
|
typedef struct UHIStat {
|
|
int16_t uhi_st_dev;
|
|
uint16_t uhi_st_ino;
|
|
uint32_t uhi_st_mode;
|
|
uint16_t uhi_st_nlink;
|
|
uint16_t uhi_st_uid;
|
|
uint16_t uhi_st_gid;
|
|
int16_t uhi_st_rdev;
|
|
uint64_t uhi_st_size;
|
|
uint64_t uhi_st_atime;
|
|
uint64_t uhi_st_spare1;
|
|
uint64_t uhi_st_mtime;
|
|
uint64_t uhi_st_spare2;
|
|
uint64_t uhi_st_ctime;
|
|
uint64_t uhi_st_spare3;
|
|
uint64_t uhi_st_blksize;
|
|
uint64_t uhi_st_blocks;
|
|
uint64_t uhi_st_spare4[2];
|
|
} UHIStat;
|
|
|
|
enum UHIOpenFlags {
|
|
UHIOpen_RDONLY = 0x0,
|
|
UHIOpen_WRONLY = 0x1,
|
|
UHIOpen_RDWR = 0x2,
|
|
UHIOpen_APPEND = 0x8,
|
|
UHIOpen_CREAT = 0x200,
|
|
UHIOpen_TRUNC = 0x400,
|
|
UHIOpen_EXCL = 0x800
|
|
};
|
|
|
|
enum UHIErrno {
|
|
UHI_EACCESS = 13,
|
|
UHI_EAGAIN = 11,
|
|
UHI_EBADF = 9,
|
|
UHI_EBADMSG = 77,
|
|
UHI_EBUSY = 16,
|
|
UHI_ECONNRESET = 104,
|
|
UHI_EEXIST = 17,
|
|
UHI_EFBIG = 27,
|
|
UHI_EINTR = 4,
|
|
UHI_EINVAL = 22,
|
|
UHI_EIO = 5,
|
|
UHI_EISDIR = 21,
|
|
UHI_ELOOP = 92,
|
|
UHI_EMFILE = 24,
|
|
UHI_EMLINK = 31,
|
|
UHI_ENAMETOOLONG = 91,
|
|
UHI_ENETDOWN = 115,
|
|
UHI_ENETUNREACH = 114,
|
|
UHI_ENFILE = 23,
|
|
UHI_ENOBUFS = 105,
|
|
UHI_ENOENT = 2,
|
|
UHI_ENOMEM = 12,
|
|
UHI_ENOSPC = 28,
|
|
UHI_ENOSR = 63,
|
|
UHI_ENOTCONN = 128,
|
|
UHI_ENOTDIR = 20,
|
|
UHI_ENXIO = 6,
|
|
UHI_EOVERFLOW = 139,
|
|
UHI_EPERM = 1,
|
|
UHI_EPIPE = 32,
|
|
UHI_ERANGE = 34,
|
|
UHI_EROFS = 30,
|
|
UHI_ESPIPE = 29,
|
|
UHI_ETIMEDOUT = 116,
|
|
UHI_ETXTBSY = 26,
|
|
UHI_EWOULDBLOCK = 11,
|
|
UHI_EXDEV = 18,
|
|
};
|
|
|
|
static void report_fault(CPUMIPSState *env)
|
|
{
|
|
int op = env->active_tc.gpr[25];
|
|
error_report("Fault during UHI operation %d", op);
|
|
abort();
|
|
}
|
|
|
|
static void uhi_cb(CPUState *cs, uint64_t ret, int err)
|
|
{
|
|
CPUMIPSState *env = cs->env_ptr;
|
|
|
|
#define E(N) case E##N: err = UHI_E##N; break
|
|
|
|
switch (err) {
|
|
case 0:
|
|
break;
|
|
E(PERM);
|
|
E(NOENT);
|
|
E(INTR);
|
|
E(BADF);
|
|
E(BUSY);
|
|
E(EXIST);
|
|
E(NOTDIR);
|
|
E(ISDIR);
|
|
E(INVAL);
|
|
E(NFILE);
|
|
E(MFILE);
|
|
E(FBIG);
|
|
E(NOSPC);
|
|
E(SPIPE);
|
|
E(ROFS);
|
|
E(NAMETOOLONG);
|
|
default:
|
|
err = UHI_EINVAL;
|
|
break;
|
|
case EFAULT:
|
|
report_fault(env);
|
|
}
|
|
|
|
#undef E
|
|
|
|
env->active_tc.gpr[2] = ret;
|
|
env->active_tc.gpr[3] = err;
|
|
}
|
|
|
|
static void uhi_fstat_cb(CPUState *cs, uint64_t ret, int err)
|
|
{
|
|
QEMU_BUILD_BUG_ON(sizeof(UHIStat) < sizeof(struct gdb_stat));
|
|
|
|
if (!err) {
|
|
CPUMIPSState *env = cs->env_ptr;
|
|
target_ulong addr = env->active_tc.gpr[5];
|
|
UHIStat *dst = lock_user(VERIFY_WRITE, addr, sizeof(UHIStat), 1);
|
|
struct gdb_stat s;
|
|
|
|
if (!dst) {
|
|
report_fault(env);
|
|
}
|
|
|
|
memcpy(&s, dst, sizeof(struct gdb_stat));
|
|
memset(dst, 0, sizeof(UHIStat));
|
|
|
|
dst->uhi_st_dev = tswap16(be32_to_cpu(s.gdb_st_dev));
|
|
dst->uhi_st_ino = tswap16(be32_to_cpu(s.gdb_st_ino));
|
|
dst->uhi_st_mode = tswap32(be32_to_cpu(s.gdb_st_mode));
|
|
dst->uhi_st_nlink = tswap16(be32_to_cpu(s.gdb_st_nlink));
|
|
dst->uhi_st_uid = tswap16(be32_to_cpu(s.gdb_st_uid));
|
|
dst->uhi_st_gid = tswap16(be32_to_cpu(s.gdb_st_gid));
|
|
dst->uhi_st_rdev = tswap16(be32_to_cpu(s.gdb_st_rdev));
|
|
dst->uhi_st_size = tswap64(be64_to_cpu(s.gdb_st_size));
|
|
dst->uhi_st_atime = tswap64(be32_to_cpu(s.gdb_st_atime));
|
|
dst->uhi_st_mtime = tswap64(be32_to_cpu(s.gdb_st_mtime));
|
|
dst->uhi_st_ctime = tswap64(be32_to_cpu(s.gdb_st_ctime));
|
|
dst->uhi_st_blksize = tswap64(be64_to_cpu(s.gdb_st_blksize));
|
|
dst->uhi_st_blocks = tswap64(be64_to_cpu(s.gdb_st_blocks));
|
|
|
|
unlock_user(dst, addr, sizeof(UHIStat));
|
|
}
|
|
|
|
uhi_cb(cs, ret, err);
|
|
}
|
|
|
|
void mips_semihosting(CPUMIPSState *env)
|
|
{
|
|
CPUState *cs = env_cpu(env);
|
|
target_ulong *gpr = env->active_tc.gpr;
|
|
const UHIOp op = gpr[25];
|
|
char *p;
|
|
|
|
switch (op) {
|
|
case UHI_exit:
|
|
gdb_exit(gpr[4]);
|
|
exit(gpr[4]);
|
|
|
|
case UHI_open:
|
|
{
|
|
target_ulong fname = gpr[4];
|
|
int ret = -1;
|
|
|
|
p = lock_user_string(fname);
|
|
if (!p) {
|
|
report_fault(env);
|
|
}
|
|
if (!strcmp("/dev/stdin", p)) {
|
|
ret = 0;
|
|
} else if (!strcmp("/dev/stdout", p)) {
|
|
ret = 1;
|
|
} else if (!strcmp("/dev/stderr", p)) {
|
|
ret = 2;
|
|
}
|
|
unlock_user(p, fname, 0);
|
|
|
|
/* FIXME: reusing a guest fd doesn't seem correct. */
|
|
if (ret >= 0) {
|
|
gpr[2] = ret;
|
|
break;
|
|
}
|
|
|
|
semihost_sys_open(cs, uhi_cb, fname, 0, gpr[5], gpr[6]);
|
|
}
|
|
break;
|
|
|
|
case UHI_close:
|
|
semihost_sys_close(cs, uhi_cb, gpr[4]);
|
|
break;
|
|
case UHI_read:
|
|
semihost_sys_read(cs, uhi_cb, gpr[4], gpr[5], gpr[6]);
|
|
break;
|
|
case UHI_write:
|
|
semihost_sys_write(cs, uhi_cb, gpr[4], gpr[5], gpr[6]);
|
|
break;
|
|
case UHI_lseek:
|
|
semihost_sys_lseek(cs, uhi_cb, gpr[4], gpr[5], gpr[6]);
|
|
break;
|
|
case UHI_unlink:
|
|
semihost_sys_remove(cs, uhi_cb, gpr[4], 0);
|
|
break;
|
|
case UHI_fstat:
|
|
semihost_sys_fstat(cs, uhi_fstat_cb, gpr[4], gpr[5]);
|
|
break;
|
|
|
|
case UHI_argc:
|
|
gpr[2] = semihosting_get_argc();
|
|
break;
|
|
case UHI_argnlen:
|
|
{
|
|
const char *s = semihosting_get_arg(gpr[4]);
|
|
gpr[2] = s ? strlen(s) : -1;
|
|
}
|
|
break;
|
|
case UHI_argn:
|
|
{
|
|
const char *s = semihosting_get_arg(gpr[4]);
|
|
target_ulong addr;
|
|
size_t len;
|
|
|
|
if (!s) {
|
|
gpr[2] = -1;
|
|
break;
|
|
}
|
|
len = strlen(s) + 1;
|
|
addr = gpr[5];
|
|
p = lock_user(VERIFY_WRITE, addr, len, 0);
|
|
if (!p) {
|
|
report_fault(env);
|
|
}
|
|
memcpy(p, s, len);
|
|
unlock_user(p, addr, len);
|
|
gpr[2] = 0;
|
|
}
|
|
break;
|
|
|
|
case UHI_plog:
|
|
{
|
|
target_ulong addr = gpr[4];
|
|
ssize_t len = target_strlen(addr);
|
|
GString *str;
|
|
char *pct_d;
|
|
|
|
if (len < 0) {
|
|
report_fault(env);
|
|
}
|
|
p = lock_user(VERIFY_READ, addr, len, 1);
|
|
if (!p) {
|
|
report_fault(env);
|
|
}
|
|
|
|
pct_d = strstr(p, "%d");
|
|
if (!pct_d) {
|
|
unlock_user(p, addr, 0);
|
|
semihost_sys_write(cs, uhi_cb, 2, addr, len);
|
|
break;
|
|
}
|
|
|
|
str = g_string_new_len(p, pct_d - p);
|
|
g_string_append_printf(str, "%d%s", (int)gpr[5], pct_d + 2);
|
|
unlock_user(p, addr, 0);
|
|
|
|
/*
|
|
* When we're using gdb, we need a guest address, so
|
|
* drop the string onto the stack below the stack pointer.
|
|
*/
|
|
if (use_gdb_syscalls()) {
|
|
addr = gpr[29] - str->len;
|
|
p = lock_user(VERIFY_WRITE, addr, str->len, 0);
|
|
if (!p) {
|
|
report_fault(env);
|
|
}
|
|
memcpy(p, str->str, str->len);
|
|
unlock_user(p, addr, str->len);
|
|
semihost_sys_write(cs, uhi_cb, 2, addr, str->len);
|
|
} else {
|
|
gpr[2] = qemu_semihosting_console_write(str->str, str->len);
|
|
}
|
|
g_string_free(str, true);
|
|
}
|
|
break;
|
|
|
|
case UHI_assert:
|
|
{
|
|
const char *msg, *file;
|
|
|
|
msg = lock_user_string(gpr[4]);
|
|
if (!msg) {
|
|
msg = "<EFAULT>";
|
|
}
|
|
file = lock_user_string(gpr[5]);
|
|
if (!file) {
|
|
file = "<EFAULT>";
|
|
}
|
|
|
|
error_report("UHI assertion \"%s\": file \"%s\", line %d",
|
|
msg, file, (int)gpr[6]);
|
|
abort();
|
|
}
|
|
|
|
default:
|
|
error_report("Unknown UHI operation %d", op);
|
|
abort();
|
|
}
|
|
return;
|
|
}
|