mirror of https://github.com/badvision/jace.git
2513 lines
88 KiB
Java
2513 lines
88 KiB
Java
// Copyright 2000-2005 the Contributors, as shown in the revision logs.
|
|
// Licensed under the Apache Public Source License 2.0 ("the License").
|
|
// You may not use this file except in compliance with the License.
|
|
|
|
package org.ibex.nestedvm;
|
|
|
|
import org.ibex.nestedvm.util.*;
|
|
import java.io.*;
|
|
import java.util.*;
|
|
import java.net.*;
|
|
import java.nio.file.*;
|
|
import java.lang.reflect.*; // For lazily linked RuntimeCompiler
|
|
|
|
// FEATURE: vfork
|
|
|
|
public abstract class UnixRuntime extends Runtime implements Cloneable {
|
|
/** The pid of this "process" */
|
|
private int pid;
|
|
private UnixRuntime parent;
|
|
public final int getPid() {
|
|
return pid;
|
|
}
|
|
|
|
private static final GlobalState defaultGS = new GlobalState();
|
|
private GlobalState gs;
|
|
public void setGlobalState(GlobalState gs) {
|
|
if(state != STOPPED) throw new IllegalStateException("can't change GlobalState when running");
|
|
if(gs == null) throw new NullPointerException("gs is null");
|
|
this.gs = gs;
|
|
}
|
|
|
|
/** proceses' current working directory - absolute path WITHOUT leading slash
|
|
"" = root, "bin" = /bin "usr/bin" = /usr/bin */
|
|
private String cwd;
|
|
|
|
/** The runtime that should be run next when in state == EXECED */
|
|
private UnixRuntime execedRuntime;
|
|
|
|
private Object children; // used only for synchronizatin
|
|
private Vector<UnixRuntime> activeChildren;
|
|
private Vector<UnixRuntime> exitedChildren;
|
|
|
|
protected UnixRuntime(int pageSize, int totalPages) {
|
|
this(pageSize,totalPages,false);
|
|
}
|
|
|
|
protected UnixRuntime(int pageSize, int totalPages, boolean exec) {
|
|
super(pageSize,totalPages,exec);
|
|
|
|
if(!exec) {
|
|
gs = defaultGS;
|
|
String userdir = Platform.getProperty("user.dir");
|
|
cwd = userdir == null ? null : gs.mapHostPath(userdir);
|
|
if(cwd == null) cwd = "/";
|
|
cwd = cwd.substring(1);
|
|
}
|
|
}
|
|
|
|
private static String posixTZ() {
|
|
StringBuffer sb = new StringBuffer();
|
|
TimeZone zone = TimeZone.getDefault();
|
|
int off = zone.getRawOffset() / 1000;
|
|
sb.append(Platform.timeZoneGetDisplayName(zone,false,false));
|
|
if(off > 0) sb.append("-");
|
|
else off = -off;
|
|
sb.append(off/3600);
|
|
off = off%3600;
|
|
if(off > 0) sb.append(":").append(off/60);
|
|
off=off%60;
|
|
if(off > 0) sb.append(":").append(off);
|
|
if(zone.useDaylightTime())
|
|
sb.append(Platform.timeZoneGetDisplayName(zone,true,false));
|
|
return sb.toString();
|
|
}
|
|
|
|
private static boolean envHas(String key,String[] environ) {
|
|
for(int i=0; i<environ.length; i++)
|
|
if(environ[i]!=null && environ[i].startsWith(key + "=")) return true;
|
|
return false;
|
|
}
|
|
|
|
String[] createEnv(String[] extra) {
|
|
String[] defaults = new String[7];
|
|
int n=0;
|
|
if(extra == null) extra = new String[0];
|
|
String tmp;
|
|
if(!envHas("USER",extra) && Platform.getProperty("user.name") != null)
|
|
defaults[n++] = "USER=" + Platform.getProperty("user.name");
|
|
if(!envHas("HOME",extra) && (tmp=Platform.getProperty("user.home")) != null && (tmp=gs.mapHostPath(tmp)) != null)
|
|
defaults[n++] = "HOME=" + tmp;
|
|
if(!envHas("TMPDIR",extra) && (tmp=Platform.getProperty("java.io.tmpdir")) != null && (tmp=gs.mapHostPath(tmp)) != null)
|
|
defaults[n++] = "TMPDIR=" + tmp;
|
|
if(!envHas("SHELL",extra)) defaults[n++] = "SHELL=/bin/sh";
|
|
if(!envHas("TERM",extra) && !win32Hacks) defaults[n++] = "TERM=vt100";
|
|
if(!envHas("TZ",extra)) defaults[n++] = "TZ=" + posixTZ();
|
|
if(!envHas("PATH",extra)) defaults[n++] = "PATH=/usr/local/bin:/usr/bin:/bin:/usr/local/sbin:/usr/sbin:/sbin";
|
|
String[] env = new String[extra.length+n];
|
|
for(int i=0; i<n; i++) env[i] = defaults[i];
|
|
for(int i=0; i<extra.length; i++) env[n++] = extra[i];
|
|
return env;
|
|
}
|
|
|
|
private static class ProcessTableFullExn extends RuntimeException { }
|
|
|
|
void _started() {
|
|
UnixRuntime[] tasks = gs.tasks;
|
|
synchronized(gs) {
|
|
if(pid != 0) {
|
|
UnixRuntime prev = tasks[pid];
|
|
if(prev == null || prev == this || prev.pid != pid || prev.parent != parent)
|
|
throw new Error("should never happen");
|
|
synchronized(parent.children) {
|
|
int i = parent.activeChildren.indexOf(prev);
|
|
if(i == -1) throw new Error("should never happen");
|
|
parent.activeChildren.setElementAt(this,i);
|
|
}
|
|
} else {
|
|
int newpid = -1;
|
|
int nextPID = gs.nextPID;
|
|
for(int i=nextPID; i<tasks.length; i++) if(tasks[i] == null) {
|
|
newpid = i;
|
|
break;
|
|
}
|
|
if(newpid == -1) for(int i=1; i<nextPID; i++) if(tasks[i] == null) {
|
|
newpid = i;
|
|
break;
|
|
}
|
|
if(newpid == -1) throw new ProcessTableFullExn();
|
|
pid = newpid;
|
|
gs.nextPID = newpid + 1;
|
|
}
|
|
tasks[pid] = this;
|
|
}
|
|
}
|
|
|
|
protected int _syscall(int syscall, int a, int b, int c, int d, int e, int f) throws ErrnoException, FaultException {
|
|
switch(syscall) {
|
|
case SYS_kill:
|
|
return sys_kill(a,b);
|
|
case SYS_fork:
|
|
return sys_fork();
|
|
case SYS_pipe:
|
|
return sys_pipe(a);
|
|
case SYS_dup2:
|
|
return sys_dup2(a,b);
|
|
case SYS_dup:
|
|
return sys_dup(a);
|
|
case SYS_waitpid:
|
|
return sys_waitpid(a,b,c);
|
|
case SYS_stat:
|
|
return sys_stat(a,b);
|
|
case SYS_lstat:
|
|
return sys_lstat(a,b);
|
|
case SYS_mkdir:
|
|
return sys_mkdir(a,b);
|
|
case SYS_getcwd:
|
|
return sys_getcwd(a,b);
|
|
case SYS_chdir:
|
|
return sys_chdir(a);
|
|
case SYS_exec:
|
|
return sys_exec(a,b,c);
|
|
case SYS_getdents:
|
|
return sys_getdents(a,b,c,d);
|
|
case SYS_unlink:
|
|
return sys_unlink(a);
|
|
case SYS_link:
|
|
return sys_link(a,b);
|
|
case SYS_getppid:
|
|
return sys_getppid();
|
|
case SYS_socket:
|
|
return sys_socket(a,b,c);
|
|
case SYS_connect:
|
|
return sys_connect(a,b,c);
|
|
case SYS_resolve_hostname:
|
|
return sys_resolve_hostname(a,b,c);
|
|
case SYS_setsockopt:
|
|
return sys_setsockopt(a,b,c,d,e);
|
|
case SYS_getsockopt:
|
|
return sys_getsockopt(a,b,c,d,e);
|
|
case SYS_getsockname:
|
|
return sys_getsockname(a,b,c);
|
|
case SYS_getpeername:
|
|
return sys_getpeername(a,b,c);
|
|
case SYS_bind:
|
|
return sys_bind(a,b,c);
|
|
case SYS_listen:
|
|
return sys_listen(a,b);
|
|
case SYS_accept:
|
|
return sys_accept(a,b,c);
|
|
case SYS_shutdown:
|
|
return sys_shutdown(a,b);
|
|
case SYS_sysctl:
|
|
return sys_sysctl(a,b,c,d,e,f);
|
|
case SYS_sendto:
|
|
return sys_sendto(a,b,c,d,e,f);
|
|
case SYS_recvfrom:
|
|
return sys_recvfrom(a,b,c,d,e,f);
|
|
case SYS_select:
|
|
return sys_select(a,b,c,d,e);
|
|
case SYS_access:
|
|
return sys_access(a,b);
|
|
case SYS_realpath:
|
|
return sys_realpath(a,b);
|
|
case SYS_chown:
|
|
return sys_chown(a,b,c);
|
|
case SYS_lchown:
|
|
return sys_chown(a,b,c);
|
|
case SYS_fchown:
|
|
return sys_fchown(a,b,c);
|
|
case SYS_chmod:
|
|
return sys_chmod(a,b,c);
|
|
case SYS_fchmod:
|
|
return sys_fchmod(a,b,c);
|
|
case SYS_fcntl:
|
|
return sys_fcntl_lock(a,b,c);
|
|
case SYS_umask:
|
|
return sys_umask(a);
|
|
case SYS_resolve_ip:
|
|
return sys_resolve_ip(a,b,c);
|
|
case SYS_rmdir:
|
|
return sys_rmdir(a);
|
|
case SYS_setsid:
|
|
return sys_setsid();
|
|
|
|
default:
|
|
return super._syscall(syscall,a,b,c,d,e,f);
|
|
}
|
|
}
|
|
|
|
|
|
int sys_resolve_ip(int addr, int ptrCBuf, int size) throws FaultException {
|
|
byte[] buffer = new byte[4];
|
|
//java.util.logging.Logger.getLogger(getClass().getName()).info("Input address = " + Integer.toString(addr));
|
|
buffer[0] = Integer.valueOf((addr >>> 24) & 0xff).byteValue();
|
|
buffer[1] = Integer.valueOf((addr >>> 16) & 0xff).byteValue();
|
|
buffer[2] = Integer.valueOf((addr >>> 8) & 0xff).byteValue();
|
|
buffer[3] = Integer.valueOf((addr >>> 0) & 0xff).byteValue();
|
|
|
|
String hostName;
|
|
try {
|
|
hostName = InetAddress.getByAddress(buffer).getHostAddress();
|
|
} catch (UnknownHostException ex) {
|
|
hostName = java.text.MessageFormat.format("{0}.{1}.{2}.{3}", new Object[] {
|
|
Integer.valueOf(buffer[0] & 0xff),
|
|
Integer.valueOf(buffer[1] & 0xff),
|
|
Integer.valueOf(buffer[2] & 0xff ),
|
|
Integer.valueOf( buffer[3] & 0xff )} );
|
|
}
|
|
//java.util.logging.Logger.getLogger(getClass().getName()).info("Output address = " + hostName);
|
|
byte[] outBytes = getBytes(hostName);
|
|
if(size == 0) return -EINVAL;
|
|
if(size < outBytes.length+1) return -ERANGE;
|
|
copyout(outBytes,ptrCBuf,outBytes.length);
|
|
memset(ptrCBuf+outBytes.length+1,0,1);
|
|
|
|
return 0;
|
|
}
|
|
|
|
FD _open(String path, int flags, int mode) throws ErrnoException {
|
|
path = normalizePath(path);
|
|
FD fd = gs.open(this,path,flags,mode);
|
|
if (fd != null && path != null) fd.setNormalizedPath(path);
|
|
return fd;
|
|
}
|
|
|
|
private int sys_getppid() {
|
|
return parent == null ? 1 : parent.pid;
|
|
}
|
|
|
|
private int sys_setsid() {
|
|
return 0;
|
|
}
|
|
|
|
private int sys_chown(int fileAddr, int uid, int gid) {
|
|
return 0;
|
|
}
|
|
|
|
private int sys_lchown(int fileAddr, int uid, int gid) {
|
|
return 0;
|
|
}
|
|
|
|
private int sys_fchown(int fd, int uid, int gid) {
|
|
return 0;
|
|
}
|
|
|
|
private int sys_chmod(int fileAddr, int uid, int gid) {
|
|
return 0;
|
|
}
|
|
|
|
private int sys_fchmod(int fd, int uid, int gid) {
|
|
return 0;
|
|
}
|
|
|
|
private int sys_umask(int mask) {
|
|
return 0;
|
|
}
|
|
|
|
private int sys_access(int cstring, int mode) throws ErrnoException, ReadFaultException {
|
|
// FEATURE: sys_access
|
|
return gs.stat(this,normalizePath(cstring(cstring))) == null ? -ENOENT : 0;
|
|
}
|
|
|
|
private int sys_realpath(int inAddr, int outAddr) throws FaultException {
|
|
String s = normalizePath(cstring(inAddr));
|
|
byte[] b = getNullTerminatedBytes(s);
|
|
if(b.length > PATH_MAX) return -ERANGE;
|
|
copyout(b,outAddr,b.length);
|
|
return 0;
|
|
}
|
|
|
|
// FEATURE: Signal handling
|
|
// check flag only on backwards jumps to basic blocks without compulsatory checks
|
|
// (see A Portable Research Framework for the Execution of Java Bytecode - Etienne Gagnon, Chapter 2)
|
|
|
|
/** The kill syscall.
|
|
SIGSTOP, SIGTSTO, SIGTTIN, and SIGTTOUT pause the process.
|
|
SIGCONT, SIGCHLD, SIGIO, and SIGWINCH are ignored.
|
|
Anything else terminates the process. */
|
|
private int sys_kill(int pid, int signal) {
|
|
// This will only be called by raise() in newlib to invoke the default handler
|
|
// We don't have to worry about actually delivering the signal
|
|
if(pid != pid) return -ESRCH;
|
|
if(signal < 0 || signal >= 32) return -EINVAL;
|
|
switch(signal) {
|
|
case 0:
|
|
return 0;
|
|
case 17: // SIGSTOP
|
|
case 18: // SIGTSTP
|
|
case 21: // SIGTTIN
|
|
case 22: // SIGTTOU
|
|
case 19: // SIGCONT
|
|
case 20: // SIGCHLD
|
|
case 23: // SIGIO
|
|
case 28: // SIGWINCH
|
|
break;
|
|
default:
|
|
exit(128+signal, true);
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
private int sys_waitpid(int pid, int statusAddr, int options) throws FaultException, ErrnoException {
|
|
final int WNOHANG = 1;
|
|
if((options & ~(WNOHANG)) != 0) return -EINVAL;
|
|
if(pid == 0 || pid < -1) {
|
|
if(STDERR_DIAG) System.err.println("WARNING: waitpid called with a pid of " + pid);
|
|
return -ECHILD;
|
|
}
|
|
boolean blocking = (options&WNOHANG)==0;
|
|
|
|
if(pid !=-1 && (pid <= 0 || pid >= gs.tasks.length)) return -ECHILD;
|
|
if(children == null) return blocking ? -ECHILD : 0;
|
|
|
|
UnixRuntime done = null;
|
|
|
|
synchronized(children) {
|
|
for(;;) {
|
|
if(pid == -1) {
|
|
if(exitedChildren.size() > 0) {
|
|
done = (UnixRuntime)exitedChildren.elementAt(exitedChildren.size() - 1);
|
|
exitedChildren.removeElementAt(exitedChildren.size() - 1);
|
|
}
|
|
} else if(pid > 0) {
|
|
if(pid >= gs.tasks.length) return -ECHILD;
|
|
UnixRuntime t = gs.tasks[pid];
|
|
if(t.parent != this) return -ECHILD;
|
|
if(t.state == EXITED) {
|
|
if(!exitedChildren.removeElement(t)) throw new Error("should never happen");
|
|
done = t;
|
|
}
|
|
} else {
|
|
// process group stuff, EINVAL returned above
|
|
throw new Error("should never happen");
|
|
}
|
|
if(done == null) {
|
|
if(!blocking) return 0;
|
|
try {
|
|
children.wait();
|
|
}
|
|
catch(InterruptedException e) {}
|
|
//System.err.println("waitpid woke up: " + exitedChildren.size());
|
|
} else {
|
|
gs.tasks[done.pid] = null;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
if(statusAddr!=0) memWrite(statusAddr,done.exitStatus()<<8);
|
|
return done.pid;
|
|
}
|
|
|
|
|
|
void _exited() {
|
|
if(children != null) synchronized(children) {
|
|
for(Enumeration e = exitedChildren.elements(); e.hasMoreElements(); ) {
|
|
UnixRuntime child = (UnixRuntime) e.nextElement();
|
|
gs.tasks[child.pid] = null;
|
|
}
|
|
exitedChildren.removeAllElements();
|
|
for(Enumeration e = activeChildren.elements(); e.hasMoreElements(); ) {
|
|
UnixRuntime child = (UnixRuntime) e.nextElement();
|
|
child.parent = null;
|
|
}
|
|
activeChildren.removeAllElements();
|
|
}
|
|
|
|
UnixRuntime _parent = parent;
|
|
if(_parent == null) {
|
|
gs.tasks[pid] = null;
|
|
} else {
|
|
synchronized(_parent.children) {
|
|
if(parent == null) {
|
|
gs.tasks[pid] = null;
|
|
} else {
|
|
if(!parent.activeChildren.removeElement(this)) throw new Error("should never happen _exited: pid: " + pid);
|
|
parent.exitedChildren.addElement(this);
|
|
parent.children.notify();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
protected Object clone() throws CloneNotSupportedException {
|
|
UnixRuntime r = (UnixRuntime) super.clone();
|
|
r.pid = 0;
|
|
r.parent = null;
|
|
r.children = null;
|
|
r.activeChildren = r.exitedChildren = null;
|
|
return r;
|
|
}
|
|
|
|
private int sys_fork() {
|
|
final UnixRuntime r;
|
|
|
|
try {
|
|
r = (UnixRuntime) clone();
|
|
} catch(Exception e) {
|
|
e.printStackTrace();
|
|
return -ENOMEM;
|
|
}
|
|
|
|
r.parent = this;
|
|
|
|
try {
|
|
r._started();
|
|
} catch(ProcessTableFullExn e) {
|
|
return -ENOMEM;
|
|
}
|
|
|
|
//System.err.println("fork " + pid + " -> " + r.pid + " tasks[" + r.pid + "] = " + gd.tasks[r.pid]);
|
|
if(children == null) {
|
|
children = new Object();
|
|
activeChildren = new Vector<UnixRuntime>();
|
|
exitedChildren = new Vector<UnixRuntime>();
|
|
}
|
|
activeChildren.addElement(r);
|
|
|
|
CPUState state = new CPUState();
|
|
getCPUState(state);
|
|
state.r[V0] = 0; // return 0 to child
|
|
state.pc += 4; // skip over syscall instruction
|
|
r.setCPUState(state);
|
|
r.state = PAUSED;
|
|
|
|
new ForkedProcess(r);
|
|
|
|
return r.pid;
|
|
}
|
|
|
|
public static final class ForkedProcess extends Thread {
|
|
private final UnixRuntime initial;
|
|
public ForkedProcess(UnixRuntime initial) {
|
|
this.initial = initial;
|
|
start();
|
|
}
|
|
public void run() {
|
|
UnixRuntime.executeAndExec(initial);
|
|
}
|
|
}
|
|
|
|
public static int runAndExec(UnixRuntime r, String argv0, String[] rest) {
|
|
return runAndExec(r,concatArgv(argv0,rest));
|
|
}
|
|
public static int runAndExec(UnixRuntime r, String[] argv) {
|
|
r.start(argv);
|
|
return executeAndExec(r);
|
|
}
|
|
|
|
public static int executeAndExec(UnixRuntime r) {
|
|
for(;;) {
|
|
for(;;) {
|
|
if(r.execute()) break;
|
|
if(STDERR_DIAG) System.err.println("WARNING: Pause requested while executing runAndExec()");
|
|
}
|
|
if(r.state != EXECED) return r.exitStatus();
|
|
r = r.execedRuntime;
|
|
}
|
|
}
|
|
|
|
private String[] readStringArray(int addr) throws ReadFaultException {
|
|
int count = 0;
|
|
for(int p=addr; memRead(p) != 0; p+=4) count++;
|
|
String[] a = new String[count];
|
|
for(int i=0,p=addr; i<count; i++,p+=4) a[i] = cstring(memRead(p));
|
|
return a;
|
|
}
|
|
|
|
private int sys_exec(int cpath, int cargv, int cenvp) throws ErrnoException, FaultException {
|
|
return exec(normalizePath(cstring(cpath)),readStringArray(cargv),readStringArray(cenvp));
|
|
}
|
|
|
|
private final static Method runtimeCompilerCompile;
|
|
static {
|
|
Method m;
|
|
try {
|
|
m = Class.forName("org.ibex.nestedvm.RuntimeCompiler").getMethod("compile",new Class[] {Seekable.class,String.class,String.class});
|
|
} catch(NoSuchMethodException e) {
|
|
m = null;
|
|
} catch(ClassNotFoundException e) {
|
|
m = null;
|
|
}
|
|
runtimeCompilerCompile = m;
|
|
}
|
|
|
|
public Class runtimeCompile(Seekable s, String sourceName) throws IOException {
|
|
if(runtimeCompilerCompile == null) {
|
|
if(STDERR_DIAG) System.err.println("WARNING: Exec attempted but RuntimeCompiler not found!");
|
|
return null;
|
|
}
|
|
|
|
try {
|
|
return (Class) runtimeCompilerCompile.invoke(null,new Object[] {s,"unixruntime,maxinsnpermethod=256,lessconstants",sourceName});
|
|
} catch(IllegalAccessException e) {
|
|
e.printStackTrace();
|
|
return null;
|
|
} catch(InvocationTargetException e) {
|
|
Throwable t = e.getTargetException();
|
|
if(t instanceof IOException) throw (IOException) t;
|
|
if(t instanceof RuntimeException) throw (RuntimeException) t;
|
|
if(t instanceof Error) throw (Error) t;
|
|
if(STDERR_DIAG) t.printStackTrace();
|
|
return null;
|
|
}
|
|
}
|
|
|
|
private int exec(String path, String[] argv, String[] envp) throws ErrnoException {
|
|
if(argv.length == 0) argv = new String[] {""};
|
|
// HACK: Hideous hack to make a standalone busybox possible
|
|
if(path.equals("bin/busybox") && getClass().getName().endsWith("BusyBox"))
|
|
return execClass(getClass(),argv,envp);
|
|
|
|
// NOTE: For this little hack to work nestedvm.root MUST be "."
|
|
/*try {
|
|
System.err.println("Execing normalized path: " + normalizedPath);
|
|
if(true) return exec(new Interpreter(normalizedPath),argv,envp);
|
|
} catch(IOException e) { throw new Error(e); }*/
|
|
|
|
FStat fstat = gs.stat(this,path);
|
|
if(fstat == null) return -ENOENT;
|
|
GlobalState.CacheEnt ent = (GlobalState.CacheEnt) gs.execCache.get(path);
|
|
long mtime = fstat.mtime();
|
|
long size = fstat.size();
|
|
if(ent != null) {
|
|
//System.err.println("Found cached entry for " + path);
|
|
if(ent.time ==mtime && ent.size == size) {
|
|
if(ent.o instanceof Class)
|
|
return execClass((Class) ent.o,argv,envp);
|
|
if(ent.o instanceof String[])
|
|
return execScript(path,(String[]) ent.o,argv,envp);
|
|
throw new Error("should never happen");
|
|
}
|
|
//System.err.println("Cache was out of date");
|
|
gs.execCache.remove(path);
|
|
}
|
|
|
|
FD fd = gs.open(this,path,RD_ONLY,0);
|
|
if(fd == null) throw new ErrnoException(ENOENT);
|
|
Seekable s = fd.seekable();
|
|
if(s == null) throw new ErrnoException(EACCES);
|
|
|
|
byte[] buf = new byte[4096];
|
|
|
|
try {
|
|
int n = s.read(buf,0,buf.length);
|
|
if(n == -1) throw new ErrnoException(ENOEXEC);
|
|
|
|
switch(buf[0]) {
|
|
case '\177': // possible ELF
|
|
if(n < 4) s.tryReadFully(buf,n,4-n);
|
|
if(buf[1] != 'E' || buf[2] != 'L' || buf[3] != 'F') return -ENOEXEC;
|
|
s.seek(0);
|
|
if(STDERR_DIAG) System.err.println("Running RuntimeCompiler for " + path);
|
|
Class c = runtimeCompile(s,path);
|
|
if(STDERR_DIAG) System.err.println("RuntimeCompiler finished for " + path);
|
|
if(c == null) throw new ErrnoException(ENOEXEC);
|
|
gs.execCache.put(path,new GlobalState.CacheEnt(mtime,size,c));
|
|
return execClass(c,argv,envp);
|
|
case '#':
|
|
if(n == 1) {
|
|
int n2 = s.read(buf,1,buf.length-1);
|
|
if(n2 == -1) return -ENOEXEC;
|
|
n += n2;
|
|
}
|
|
if(buf[1] != '!') return -ENOEXEC;
|
|
int p = 2;
|
|
n -= 2;
|
|
OUTER: for(;;) {
|
|
for(int i=p; i<p+n; i++) if(buf[i] == '\n') {
|
|
p = i;
|
|
break OUTER;
|
|
}
|
|
p += n;
|
|
if(p == buf.length) break OUTER;
|
|
n = s.read(buf,p,buf.length-p);
|
|
}
|
|
int cmdStart = 2;
|
|
for(; cmdStart<p; cmdStart++) if(buf[cmdStart]!=' ') break;
|
|
if(cmdStart == p) throw new ErrnoException(ENOEXEC);
|
|
int argStart = cmdStart;
|
|
for(; argStart<p; argStart++) if(buf[argStart] == ' ') break;
|
|
int cmdEnd = argStart;
|
|
while(argStart < p && buf[argStart] == ' ') argStart++;
|
|
String[] command = new String[] {
|
|
new String(buf,cmdStart,cmdEnd-cmdStart),
|
|
argStart < p ? new String(buf,argStart,p-argStart) : null
|
|
};
|
|
gs.execCache.put(path,new GlobalState.CacheEnt(mtime,size,command));
|
|
return execScript(path,command,argv,envp);
|
|
default:
|
|
return -ENOEXEC;
|
|
}
|
|
} catch(IOException e) {
|
|
return -EIO;
|
|
} finally {
|
|
fd.close();
|
|
}
|
|
}
|
|
|
|
public int execScript(String path, String[] command, String[] argv, String[] envp) throws ErrnoException {
|
|
String[] newArgv = new String[argv.length-1 + (command[1] != null ? 3 : 2)];
|
|
int p = command[0].lastIndexOf('/');
|
|
newArgv[0] = p == -1 ? command[0] : command[0].substring(p+1);
|
|
newArgv[1] = "/" + path;
|
|
p = 2;
|
|
if(command[1] != null) newArgv[p++] = command[1];
|
|
for(int i=1; i<argv.length; i++) newArgv[p++] = argv[i];
|
|
if(p != newArgv.length) throw new Error("p != newArgv.length");
|
|
System.err.println("Execing: " + command[0]);
|
|
for(int i=0; i<newArgv.length; i++) System.err.println("execing [" + i + "] " + newArgv[i]);
|
|
return exec(command[0],newArgv,envp);
|
|
}
|
|
|
|
public int execClass(Class<?> c,String[] argv, String[] envp) {
|
|
try {
|
|
UnixRuntime r = (UnixRuntime) c.getDeclaredConstructor(new Class[] {Boolean.TYPE}).newInstance(new Object[] {Boolean.TRUE});
|
|
return exec(r,argv,envp);
|
|
} catch(Exception e) {
|
|
e.printStackTrace();
|
|
return -ENOEXEC;
|
|
}
|
|
}
|
|
|
|
private int exec(UnixRuntime r, String[] argv, String[] envp) {
|
|
//System.err.println("Execing " + r);
|
|
for(int i=0; i<OPEN_MAX; i++) if(closeOnExec[i]) closeFD(i);
|
|
r.fds = fds;
|
|
r.closeOnExec = closeOnExec;
|
|
// make sure this doesn't get messed with these since we didn't copy them
|
|
fds = null;
|
|
closeOnExec = null;
|
|
|
|
r.gs = gs;
|
|
r.sm = sm;
|
|
r.cwd = cwd;
|
|
r.pid = pid;
|
|
r.parent = parent;
|
|
r.start(argv,envp);
|
|
|
|
state = EXECED;
|
|
execedRuntime = r;
|
|
|
|
return 0;
|
|
}
|
|
|
|
public static class Pipe {
|
|
private final byte[] pipebuf = new byte[PIPE_BUF*4];
|
|
private int readPos;
|
|
private int writePos;
|
|
|
|
public final FD reader = new Reader();
|
|
public final FD writer = new Writer();
|
|
|
|
public class Reader extends FD {
|
|
protected FStat _fstat() { return new SocketFStat(); }
|
|
public int read(byte[] buf, int off, int len) throws ErrnoException {
|
|
if(len == 0) return 0;
|
|
synchronized(Pipe.this) {
|
|
while(writePos != -1 && readPos == writePos) {
|
|
try {
|
|
Pipe.this.wait();
|
|
} catch(InterruptedException e) {
|
|
/* ignore */
|
|
}
|
|
}
|
|
if(writePos == -1) return 0; // eof
|
|
len = Math.min(len,writePos-readPos);
|
|
System.arraycopy(pipebuf,readPos,buf,off,len);
|
|
readPos += len;
|
|
if(readPos == writePos) Pipe.this.notify();
|
|
return len;
|
|
}
|
|
}
|
|
public int flags() {
|
|
return O_RDONLY;
|
|
}
|
|
public void _close() {
|
|
synchronized(Pipe.this) {
|
|
readPos = -1;
|
|
Pipe.this.notify();
|
|
}
|
|
}
|
|
}
|
|
|
|
public class Writer extends FD {
|
|
protected FStat _fstat() {
|
|
return new SocketFStat();
|
|
}
|
|
public int write(byte[] buf, int off, int len) throws ErrnoException {
|
|
if(len == 0) return 0;
|
|
synchronized(Pipe.this) {
|
|
if(readPos == -1) throw new ErrnoException(EPIPE);
|
|
if(pipebuf.length - writePos < Math.min(len,PIPE_BUF)) {
|
|
// not enough space to atomicly write the data
|
|
while(readPos != -1 && readPos != writePos) {
|
|
try {
|
|
Pipe.this.wait();
|
|
}
|
|
catch(InterruptedException e) {
|
|
/* ignore */
|
|
}
|
|
}
|
|
if(readPos == -1) throw new ErrnoException(EPIPE);
|
|
readPos = writePos = 0;
|
|
}
|
|
len = Math.min(len,pipebuf.length - writePos);
|
|
System.arraycopy(buf,off,pipebuf,writePos,len);
|
|
if(readPos == writePos) Pipe.this.notify();
|
|
writePos += len;
|
|
return len;
|
|
}
|
|
}
|
|
public int flags() {
|
|
return O_WRONLY;
|
|
}
|
|
public void _close() {
|
|
synchronized(Pipe.this) {
|
|
writePos = -1;
|
|
Pipe.this.notify();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
private int sys_pipe(int addr) {
|
|
Pipe pipe = new Pipe();
|
|
|
|
int fd1 = addFD(pipe.reader);
|
|
if(fd1 < 0) return -ENFILE;
|
|
int fd2 = addFD(pipe.writer);
|
|
if(fd2 < 0) {
|
|
closeFD(fd1);
|
|
return -ENFILE;
|
|
}
|
|
|
|
try {
|
|
memWrite(addr,fd1);
|
|
memWrite(addr+4,fd2);
|
|
} catch(FaultException e) {
|
|
closeFD(fd1);
|
|
closeFD(fd2);
|
|
return -EFAULT;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
private int sys_dup2(int oldd, int newd) {
|
|
if(oldd == newd) return 0;
|
|
if(oldd < 0 || oldd >= OPEN_MAX) return -EBADFD;
|
|
if(newd < 0 || newd >= OPEN_MAX) return -EBADFD;
|
|
if(fds[oldd] == null) return -EBADFD;
|
|
if(fds[newd] != null) fds[newd].close();
|
|
fds[newd] = fds[oldd].dup();
|
|
return 0;
|
|
}
|
|
|
|
private int sys_dup(int oldd) {
|
|
if(oldd < 0 || oldd >= OPEN_MAX) return -EBADFD;
|
|
if(fds[oldd] == null) return -EBADFD;
|
|
FD fd = fds[oldd].dup();
|
|
int newd = addFD(fd);
|
|
if(newd < 0) {
|
|
fd.close();
|
|
return -ENFILE;
|
|
}
|
|
return newd;
|
|
}
|
|
|
|
private int sys_stat(int cstring, int addr) throws FaultException, ErrnoException {
|
|
FStat s = gs.stat(this,normalizePath(cstring(cstring)));
|
|
if(s == null) return -ENOENT;
|
|
return stat(s,addr);
|
|
}
|
|
|
|
private int sys_lstat(int cstring, int addr) throws FaultException, ErrnoException {
|
|
FStat s = gs.lstat(this,normalizePath(cstring(cstring)));
|
|
if(s == null) return -ENOENT;
|
|
return stat(s,addr);
|
|
}
|
|
|
|
private int sys_mkdir(int cstring, int mode) throws FaultException, ErrnoException {
|
|
gs.mkdir(this,normalizePath(cstring(cstring)),mode);
|
|
return 0;
|
|
}
|
|
|
|
private int sys_unlink(int cstring) throws FaultException, ErrnoException {
|
|
gs.unlink(this,normalizePath(cstring(cstring)));
|
|
return 0;
|
|
}
|
|
|
|
private int sys_link(int cstring1, int cstring2) throws FaultException, ErrnoException {
|
|
gs.link(this,normalizePath(cstring(cstring1)), normalizePath(cstring(cstring2)));
|
|
return 0;
|
|
}
|
|
|
|
private int sys_rmdir(int ptrPathName) throws ErrnoException, ReadFaultException {
|
|
gs.rmdir(this, normalizePath(cstring(ptrPathName)));
|
|
return 0;
|
|
}
|
|
|
|
private int sys_getcwd(int addr, int size) throws FaultException, ErrnoException {
|
|
byte[] b = getBytes(cwd);
|
|
if(size == 0) return -EINVAL;
|
|
if(size < b.length+2) return -ERANGE;
|
|
memset(addr,'/',1);
|
|
copyout(b,addr+1,b.length);
|
|
memset(addr+b.length+1,0,1);
|
|
return addr;
|
|
}
|
|
|
|
private int sys_chdir(int addr) throws ErrnoException, FaultException {
|
|
String path = normalizePath(cstring(addr));
|
|
FStat st = gs.stat(this,path);
|
|
if(st == null) return -ENOENT;
|
|
if(st.type() != FStat.S_IFDIR) return -ENOTDIR;
|
|
cwd = path;
|
|
return 0;
|
|
}
|
|
|
|
private int sys_getdents(int fdn, int addr, int count, int seekptr) throws FaultException, ErrnoException {
|
|
count = Math.min(count,MAX_CHUNK);
|
|
if(fdn < 0 || fdn >= OPEN_MAX) return -EBADFD;
|
|
if(fds[fdn] == null) return -EBADFD;
|
|
byte[] buf = byteBuf(count);
|
|
int n = fds[fdn].getdents(buf,0,count);
|
|
copyout(buf,addr,n);
|
|
return n;
|
|
}
|
|
|
|
void _preCloseFD(FD fd) {
|
|
// release all fcntl locks on this file
|
|
Seekable s = fd.seekable();
|
|
if (s == null) return;
|
|
|
|
try {
|
|
for (int i=0; i < gs.locks.length; i++) {
|
|
Seekable.Lock l = gs.locks[i];
|
|
if (l == null) continue;
|
|
if (s.equals(l.seekable()) && l.getOwner() == this) {
|
|
l.release();
|
|
gs.locks[i] = null;
|
|
}
|
|
}
|
|
} catch (IOException e) {
|
|
throw new RuntimeException(e);
|
|
}
|
|
}
|
|
|
|
void _postCloseFD(FD fd) {
|
|
if (fd.isMarkedForDeleteOnClose()) {
|
|
try {
|
|
gs.unlink(this, fd.getNormalizedPath());
|
|
}
|
|
catch (Throwable t) {}
|
|
}
|
|
}
|
|
|
|
/** Implements the F_GETLK and F_SETLK cases of fcntl syscall.
|
|
* If l_start = 0 and l_len = 0 the lock refers to the entire file.
|
|
* Uses GlobalState to ensure locking across processes in the same JVM.
|
|
struct flock {
|
|
short l_type; // lock type: F_UNLCK, F_RDLCK, F_WRLCK
|
|
short l_whence; // type of l_start: SEEK_SET, SEEK_CUR, SEEK_END
|
|
long l_start; // starting offset, bytes
|
|
long l_len; // len = 0 means until EOF
|
|
short l_pid; // lock owner
|
|
short l_xxx; // padding
|
|
};
|
|
*/
|
|
private int sys_fcntl_lock(int fdn, int cmd, int arg) throws FaultException {
|
|
if (cmd != F_GETLK && cmd != F_SETLK) return sys_fcntl(fdn, cmd, arg);
|
|
|
|
if(fdn < 0 || fdn >= OPEN_MAX) return -EBADFD;
|
|
if(fds[fdn] == null) return -EBADFD;
|
|
FD fd = fds[fdn];
|
|
|
|
if (arg == 0) return -EINVAL;
|
|
int word = memRead(arg);
|
|
int l_start = memRead(arg+4);
|
|
int l_len = memRead(arg+8);
|
|
int l_type = word>>16;
|
|
int l_whence = word&0x00ff;
|
|
|
|
Seekable.Lock[] locks = gs.locks;
|
|
Seekable s = fd.seekable();
|
|
if (s == null) return -EINVAL;
|
|
|
|
try {
|
|
|
|
switch (l_whence) {
|
|
case SEEK_SET:
|
|
break;
|
|
case SEEK_CUR:
|
|
l_start += s.pos();
|
|
break;
|
|
case SEEK_END:
|
|
l_start += s.length();
|
|
break;
|
|
default:
|
|
return -1;
|
|
}
|
|
|
|
if (cmd == F_GETLK) {
|
|
// The simple Java file locking below will happily return
|
|
// a lock that overlaps one already held by the JVM. Thus
|
|
// we must check over all the locks held by other Runtimes
|
|
for (int i=0; i < locks.length; i++) {
|
|
if (locks[i] == null || !s.equals(locks[i].seekable()))
|
|
continue;
|
|
if (!locks[i].overlaps(l_start, l_len))
|
|
continue;
|
|
if (locks[i].getOwner() == this)
|
|
continue;
|
|
if (locks[i].isShared() && l_type == F_RDLCK)
|
|
continue;
|
|
|
|
// overlapping lock held by another process
|
|
return 0;
|
|
}
|
|
|
|
// check if an area is lockable by attempting to obtain a lock
|
|
Seekable.Lock lock = s.lock(l_start, l_len, l_type == F_RDLCK);
|
|
|
|
if (lock != null) { // no lock exists
|
|
memWrite(arg, SEEK_SET|(F_UNLCK<<16));
|
|
lock.release();
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
// now processing F_SETLK
|
|
if (cmd != F_SETLK) return -EINVAL;
|
|
|
|
if (l_type == F_UNLCK) {
|
|
// release all locks that fall within the boundaries given
|
|
for (int i=0; i < locks.length; i++) {
|
|
if (locks[i] == null || !s.equals(locks[i].seekable()))
|
|
continue;
|
|
if (locks[i].getOwner() != this) continue;
|
|
|
|
int pos = (int)locks[i].position();
|
|
if (pos < l_start) continue;
|
|
if (l_start != 0 && l_len != 0) // start/len 0 means unlock all
|
|
if (pos + locks[i].size() > l_start + l_len)
|
|
continue;
|
|
|
|
locks[i].release();
|
|
locks[i] = null;
|
|
}
|
|
return 0;
|
|
|
|
} else if (l_type == F_RDLCK || l_type == F_WRLCK) {
|
|
// first see if a lock already exists
|
|
for (int i=0; i < locks.length; i++) {
|
|
if (locks[i] == null || !s.equals(locks[i].seekable()))
|
|
continue;
|
|
|
|
if (locks[i].getOwner() == this) {
|
|
// if this Runtime owns an overlapping lock work with it
|
|
if (locks[i].contained(l_start, l_len)) {
|
|
locks[i].release();
|
|
locks[i] = null;
|
|
} else if (locks[i].contains(l_start, l_len)) {
|
|
if (locks[i].isShared() == (l_type == F_RDLCK)) {
|
|
// return this more general lock
|
|
memWrite(arg+4, (int)locks[i].position());
|
|
memWrite(arg+8, (int)locks[i].size());
|
|
return 0;
|
|
} else {
|
|
locks[i].release();
|
|
locks[i] = null;
|
|
}
|
|
}
|
|
} else {
|
|
// if another Runtime has an lock and it is exclusive or
|
|
// we want an exclusive lock then fail
|
|
if (locks[i].overlaps(l_start, l_len)
|
|
&& (!locks[i].isShared() || l_type == F_WRLCK))
|
|
return -EAGAIN;
|
|
}
|
|
}
|
|
|
|
// create the lock
|
|
Seekable.Lock lock = s.lock(l_start, l_len, l_type == F_RDLCK);
|
|
if (lock == null) return -EAGAIN;
|
|
lock.setOwner(this);
|
|
|
|
int i;
|
|
for (i=0; i < locks.length; i++)
|
|
if (locks[i] == null) break;
|
|
if (i == locks.length) return -ENOLCK;
|
|
locks[i] = lock;
|
|
return 0;
|
|
|
|
} else {
|
|
return -EINVAL;
|
|
}
|
|
|
|
} catch (IOException e) {
|
|
throw new RuntimeException(e);
|
|
}
|
|
}
|
|
|
|
static class SocketFD extends FD {
|
|
public static final int TYPE_STREAM = 0;
|
|
public static final int TYPE_DGRAM = 1;
|
|
public static final int LISTEN = 2;
|
|
public int type() {
|
|
return flags & 1;
|
|
}
|
|
public boolean listen() {
|
|
return (flags & 2) != 0;
|
|
}
|
|
|
|
int flags;
|
|
int options;
|
|
|
|
Socket s;
|
|
ServerSocket ss;
|
|
DatagramSocket ds;
|
|
|
|
InetAddress bindAddr;
|
|
int bindPort = -1;
|
|
InetAddress connectAddr;
|
|
int connectPort = -1;
|
|
|
|
DatagramPacket dp;
|
|
InputStream is;
|
|
OutputStream os;
|
|
|
|
private static final byte[] EMPTY = new byte[0];
|
|
public SocketFD(int type) {
|
|
flags = type;
|
|
if(type == TYPE_DGRAM)
|
|
dp = new DatagramPacket(EMPTY,0);
|
|
}
|
|
|
|
public SocketAddress getLocalSocketAddress() {
|
|
int type = type();
|
|
if(type == SocketFD.TYPE_STREAM) {
|
|
if(listen()) {
|
|
return ss.getLocalSocketAddress();
|
|
} else {
|
|
return s.getLocalSocketAddress();
|
|
}
|
|
} else {
|
|
return ds.getLocalSocketAddress();
|
|
}
|
|
}
|
|
|
|
public void setOptions() {
|
|
try {
|
|
if(s != null && type() == TYPE_STREAM && !listen()) {
|
|
Platform.socketSetKeepAlive(s,(options & SO_KEEPALIVE) != 0);
|
|
}
|
|
} catch(SocketException e) {
|
|
if(STDERR_DIAG) e.printStackTrace();
|
|
}
|
|
}
|
|
|
|
public void _close() {
|
|
try {
|
|
if(s != null) s.close();
|
|
if(ss != null) ss.close();
|
|
if(ds != null) ds.close();
|
|
} catch(IOException e) {
|
|
/* ignore */
|
|
}
|
|
}
|
|
|
|
public int read(byte[] a, int off, int length) throws ErrnoException {
|
|
if(type() == TYPE_DGRAM) return recvfrom(a,off,length,null,null);
|
|
if(is == null) throw new ErrnoException(EPIPE);
|
|
try {
|
|
int n = is.read(a,off,length);
|
|
return n < 0 ? 0 : n;
|
|
} catch(IOException e) {
|
|
throw new ErrnoException(EIO);
|
|
}
|
|
}
|
|
|
|
public int recvfrom(byte[] a, int off, int length, InetAddress[] sockAddr, int[] port) throws ErrnoException {
|
|
if(type() == TYPE_STREAM) return read(a,off,length);
|
|
|
|
if(off != 0) throw new IllegalArgumentException("off must be 0");
|
|
dp.setData(a);
|
|
dp.setLength(length);
|
|
try {
|
|
if(ds == null) ds = new DatagramSocket();
|
|
ds.receive(dp);
|
|
} catch(IOException e) {
|
|
if(STDERR_DIAG) e.printStackTrace();
|
|
throw new ErrnoException(EIO);
|
|
}
|
|
if(sockAddr != null) {
|
|
sockAddr[0] = dp.getAddress();
|
|
port[0] = dp.getPort();
|
|
}
|
|
return dp.getLength();
|
|
}
|
|
|
|
public int write(byte[] a, int off, int length) throws ErrnoException {
|
|
if(type() == TYPE_DGRAM) return sendto(a,off,length,null,-1);
|
|
|
|
if(os == null) throw new ErrnoException(EPIPE);
|
|
try {
|
|
os.write(a,off,length);
|
|
return length;
|
|
} catch(IOException e) {
|
|
throw new ErrnoException(EIO);
|
|
}
|
|
}
|
|
|
|
public int sendto(byte[] a, int off, int length, InetAddress destAddr, int destPort) throws ErrnoException {
|
|
if(off != 0) throw new IllegalArgumentException("off must be 0");
|
|
if(type() == TYPE_STREAM) return write(a,off,length);
|
|
|
|
if(destAddr == null) {
|
|
destAddr = connectAddr;
|
|
destPort = connectPort;
|
|
|
|
if(destAddr == null) throw new ErrnoException(ENOTCONN);
|
|
}
|
|
|
|
dp.setAddress(destAddr);
|
|
dp.setPort(destPort);
|
|
dp.setData(a);
|
|
dp.setLength(length);
|
|
|
|
try {
|
|
if(ds == null) ds = new DatagramSocket();
|
|
ds.send(dp);
|
|
} catch(IOException e) {
|
|
if(STDERR_DIAG) e.printStackTrace();
|
|
if("Network is unreachable".equals(e.getMessage())) throw new ErrnoException(EHOSTUNREACH);
|
|
throw new ErrnoException(EIO);
|
|
}
|
|
return dp.getLength();
|
|
}
|
|
|
|
public int flags() {
|
|
return O_RDWR;
|
|
}
|
|
public FStat _fstat() {
|
|
return new SocketFStat();
|
|
}
|
|
}
|
|
|
|
private int sys_socket(int domain, int type, int proto) {
|
|
if(domain != AF_INET || (type != SOCK_STREAM && type != SOCK_DGRAM)) return -EPROTONOSUPPORT;
|
|
return addFD(new SocketFD(type == SOCK_STREAM ? SocketFD.TYPE_STREAM : SocketFD.TYPE_DGRAM));
|
|
}
|
|
|
|
private SocketFD getSocketFD(int fdn) throws ErrnoException {
|
|
if(fdn < 0 || fdn >= OPEN_MAX) throw new ErrnoException(EBADFD);
|
|
if(fds[fdn] == null) throw new ErrnoException(EBADFD);
|
|
if(!(fds[fdn] instanceof SocketFD)) throw new ErrnoException(ENOTSOCK);
|
|
|
|
return (SocketFD) fds[fdn];
|
|
}
|
|
|
|
private int sys_connect(int fdn, int addr, int namelen) throws ErrnoException, FaultException {
|
|
SocketFD fd = getSocketFD(fdn);
|
|
|
|
if(fd.type() == SocketFD.TYPE_STREAM && (fd.s != null || fd.ss != null)) return -EISCONN;
|
|
int word1 = memRead(addr);
|
|
if( ((word1 >>> 16)&0xff) != AF_INET) return -EAFNOSUPPORT;
|
|
int port = word1 & 0xffff;
|
|
byte[] ip = new byte[4];
|
|
copyin(addr+4,ip,4);
|
|
|
|
InetAddress inetAddr;
|
|
try {
|
|
inetAddr = Platform.inetAddressFromBytes(ip);
|
|
} catch(UnknownHostException e) {
|
|
return -EADDRNOTAVAIL;
|
|
}
|
|
|
|
fd.connectAddr = inetAddr;
|
|
fd.connectPort = port;
|
|
|
|
try {
|
|
switch(fd.type()) {
|
|
case SocketFD.TYPE_STREAM: {
|
|
Socket s = new Socket(inetAddr,port);
|
|
fd.s = s;
|
|
fd.setOptions();
|
|
fd.is = s.getInputStream();
|
|
fd.os = s.getOutputStream();
|
|
break;
|
|
}
|
|
case SocketFD.TYPE_DGRAM:
|
|
break;
|
|
default:
|
|
throw new Error("should never happen");
|
|
}
|
|
} catch(IOException e) {
|
|
return -ECONNREFUSED;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
private int sys_resolve_hostname(int chostname, int addr, int sizeAddr) throws FaultException {
|
|
String hostname = cstring(chostname);
|
|
int size = memRead(sizeAddr);
|
|
InetAddress[] inetAddrs;
|
|
try {
|
|
inetAddrs = InetAddress.getAllByName(hostname);
|
|
} catch(UnknownHostException e) {
|
|
return HOST_NOT_FOUND;
|
|
}
|
|
int count = min(size/4,inetAddrs.length);
|
|
for(int i=0; i<count; i++,addr+=4) {
|
|
byte[] b = inetAddrs[i].getAddress();
|
|
copyout(b,addr,4);
|
|
}
|
|
memWrite(sizeAddr,count*4);
|
|
return 0;
|
|
}
|
|
|
|
private int sys_setsockopt(int fdn, int level, int name, int valaddr, int len) throws ReadFaultException, ErrnoException {
|
|
SocketFD fd = getSocketFD(fdn);
|
|
switch(level) {
|
|
case SOL_SOCKET:
|
|
switch(name) {
|
|
case SO_REUSEADDR:
|
|
case SO_KEEPALIVE: {
|
|
if(len != 4) return -EINVAL;
|
|
int val = memRead(valaddr);
|
|
if(val != 0) fd.options |= name;
|
|
else fd.options &= ~name;
|
|
fd.setOptions();
|
|
return 0;
|
|
}
|
|
default:
|
|
if(STDERR_DIAG) System.err.println("Unknown setsockopt name passed: " + name);
|
|
return -ENOPROTOOPT;
|
|
}
|
|
default:
|
|
if(STDERR_DIAG) System.err.println("Unknown setsockopt leve passed: " + level);
|
|
return -ENOPROTOOPT;
|
|
}
|
|
}
|
|
|
|
private int sys_getpeername(int socketDescriptor, int ptrSockName, int ptrNameLength) throws FaultException, ErrnoException
|
|
{
|
|
SocketFD fd = getSocketFD(socketDescriptor);
|
|
InetSocketAddress isa = (InetSocketAddress) fd.s.getRemoteSocketAddress();
|
|
return getfooonamehelper(isa, ptrSockName, ptrNameLength);
|
|
}
|
|
|
|
private int sys_getsockname(int socketDescriptor, int ptrSockName, int ptrNameLength) throws FaultException, ErrnoException
|
|
{
|
|
SocketFD fd = getSocketFD(socketDescriptor);
|
|
InetSocketAddress isa = (InetSocketAddress) fd.getLocalSocketAddress();
|
|
return getfooonamehelper(isa, ptrSockName, ptrNameLength);
|
|
}
|
|
|
|
|
|
private int getfooonamehelper(InetSocketAddress isa, int ptrSockName, int ptrNameLength) throws FaultException, ErrnoException
|
|
{
|
|
int length = 8;
|
|
int bufferSize = memRead(ptrNameLength);
|
|
byte[] ia = isa.getAddress().getAddress();
|
|
int port = isa.getPort();
|
|
|
|
/*
|
|
byte sin_len
|
|
byte sin_family
|
|
short sin_port
|
|
int sin_addr
|
|
*/
|
|
int dword1 = ((length & 0xff) << 24) | ((AF_INET & 0xff) << 16) | ( (port & 0xffff) << 0);
|
|
int dword2 = (ia[0] << 24) | (ia[1] << 16) | (ia[2] << 8) | (ia[3] << 0);
|
|
memWrite(ptrNameLength, length);
|
|
if (bufferSize == 0) {
|
|
return 0;
|
|
}
|
|
if (bufferSize < 4) {
|
|
int oldDword1 = memRead(ptrSockName);
|
|
int lowermask = (1 << bufferSize * 8) - 1;
|
|
int highermask = -1 - lowermask;
|
|
dword1 = (oldDword1 & highermask) | (dword1 & lowermask);
|
|
}
|
|
memWrite(ptrSockName, dword1);
|
|
if (bufferSize < 8) {
|
|
int oldDword2 = memRead(ptrSockName+4);
|
|
int lowermask = (1 << (bufferSize - 4) * 8) - 1;
|
|
int highermask = -1 - lowermask;
|
|
dword2 = (oldDword2 & highermask) | (dword2 & lowermask);
|
|
}
|
|
memWrite(ptrSockName + 4, dword2);
|
|
return 0;
|
|
}
|
|
|
|
private int sys_getsockopt(int fdn, int level, int name, int valaddr, int lenaddr) throws ErrnoException, FaultException {
|
|
SocketFD fd = getSocketFD(fdn);
|
|
switch(level) {
|
|
case SOL_SOCKET:
|
|
switch(name) {
|
|
case SO_REUSEADDR:
|
|
case SO_KEEPALIVE: {
|
|
int len = memRead(lenaddr);
|
|
if(len < 4) return -EINVAL;
|
|
int val = (fd.options & name) != 0 ? 1 : 0;
|
|
memWrite(valaddr,val);
|
|
memWrite(lenaddr,4);
|
|
return 0;
|
|
}
|
|
default:
|
|
if(STDERR_DIAG) System.err.println("Unknown setsockopt name passed: " + name);
|
|
return -ENOPROTOOPT;
|
|
}
|
|
default:
|
|
if(STDERR_DIAG) System.err.println("Unknown setsockopt leve passed: " + level);
|
|
return -ENOPROTOOPT;
|
|
}
|
|
}
|
|
|
|
private int sys_bind(int fdn, int addr, int namelen) throws FaultException, ErrnoException {
|
|
SocketFD fd = getSocketFD(fdn);
|
|
|
|
if(fd.type() == SocketFD.TYPE_STREAM && (fd.s != null || fd.ss != null)) return -EISCONN;
|
|
int word1 = memRead(addr);
|
|
if( ((word1 >>> 16)&0xff) != AF_INET) return -EAFNOSUPPORT;
|
|
int port = word1 & 0xffff;
|
|
InetAddress inetAddr = null;
|
|
if(memRead(addr+4) != 0) {
|
|
byte[] ip = new byte[4];
|
|
copyin(addr+4,ip,4);
|
|
|
|
try {
|
|
inetAddr = Platform.inetAddressFromBytes(ip);
|
|
} catch(UnknownHostException e) {
|
|
return -EADDRNOTAVAIL;
|
|
}
|
|
}
|
|
|
|
switch(fd.type()) {
|
|
case SocketFD.TYPE_STREAM: {
|
|
fd.bindAddr = inetAddr;
|
|
fd.bindPort = port;
|
|
return 0;
|
|
}
|
|
case SocketFD.TYPE_DGRAM: {
|
|
if(fd.ds != null) fd.ds.close();
|
|
try {
|
|
fd.ds = inetAddr != null ? new DatagramSocket(port,inetAddr) : new DatagramSocket(port);
|
|
} catch(IOException e) {
|
|
return -EADDRINUSE;
|
|
}
|
|
return 0;
|
|
}
|
|
default:
|
|
throw new Error("should never happen");
|
|
}
|
|
}
|
|
|
|
private int sys_listen(int fdn, int backlog) throws ErrnoException {
|
|
SocketFD fd = getSocketFD(fdn);
|
|
if(fd.type() != SocketFD.TYPE_STREAM) return -EOPNOTSUPP;
|
|
if(fd.ss != null || fd.s != null) return -EISCONN;
|
|
if(fd.bindPort < 0) return -EOPNOTSUPP;
|
|
|
|
try {
|
|
fd.ss = new ServerSocket(fd.bindPort,backlog,fd.bindAddr);
|
|
fd.flags |= SocketFD.LISTEN;
|
|
return 0;
|
|
} catch(IOException e) {
|
|
return -EADDRINUSE;
|
|
}
|
|
|
|
}
|
|
|
|
private int sys_accept(int fdn, int addr, int lenaddr) throws ErrnoException, FaultException {
|
|
SocketFD fd = getSocketFD(fdn);
|
|
if(fd.type() != SocketFD.TYPE_STREAM) return -EOPNOTSUPP;
|
|
if(!fd.listen()) return -EOPNOTSUPP;
|
|
|
|
int size = memRead(lenaddr);
|
|
|
|
ServerSocket s = fd.ss;
|
|
Socket client;
|
|
try {
|
|
client = s.accept();
|
|
} catch(IOException e) {
|
|
return -EIO;
|
|
}
|
|
|
|
if(size >= 8) {
|
|
memWrite(addr,(6 << 24) | (AF_INET << 16) | client.getPort());
|
|
byte[] b = client.getInetAddress().getAddress();
|
|
copyout(b,addr+4,4);
|
|
memWrite(lenaddr,8);
|
|
}
|
|
|
|
SocketFD clientFD = new SocketFD(SocketFD.TYPE_STREAM);
|
|
clientFD.s = client;
|
|
try {
|
|
clientFD.is = client.getInputStream();
|
|
clientFD.os = client.getOutputStream();
|
|
} catch(IOException e) {
|
|
return -EIO;
|
|
}
|
|
int n = addFD(clientFD);
|
|
if(n == -1) {
|
|
clientFD.close();
|
|
return -ENFILE;
|
|
}
|
|
return n;
|
|
}
|
|
|
|
private int sys_shutdown(int fdn, int how) throws ErrnoException {
|
|
SocketFD fd = getSocketFD(fdn);
|
|
if(fd.type() != SocketFD.TYPE_STREAM || fd.listen()) return -EOPNOTSUPP;
|
|
if(fd.s == null) return -ENOTCONN;
|
|
|
|
Socket s = fd.s;
|
|
|
|
try {
|
|
if(how == SHUT_RD || how == SHUT_RDWR) Platform.socketHalfClose(s,false);
|
|
if(how == SHUT_WR || how == SHUT_RDWR) Platform.socketHalfClose(s,true);
|
|
} catch(IOException e) {
|
|
return -EIO;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
private int sys_sendto(int fdn, int addr, int count, int flags, int destAddr, int socklen) throws ErrnoException,ReadFaultException {
|
|
SocketFD fd = getSocketFD(fdn);
|
|
if(flags != 0) throw new ErrnoException(EINVAL);
|
|
|
|
int word1 = memRead(destAddr);
|
|
if( ((word1 >>> 16)&0xff) != AF_INET) return -EAFNOSUPPORT;
|
|
int port = word1 & 0xffff;
|
|
InetAddress inetAddr;
|
|
byte[] ip = new byte[4];
|
|
copyin(destAddr+4,ip,4);
|
|
try {
|
|
inetAddr = Platform.inetAddressFromBytes(ip);
|
|
} catch(UnknownHostException e) {
|
|
return -EADDRNOTAVAIL;
|
|
}
|
|
|
|
count = Math.min(count,MAX_CHUNK);
|
|
byte[] buf = byteBuf(count);
|
|
copyin(addr,buf,count);
|
|
try {
|
|
return fd.sendto(buf,0,count,inetAddr,port);
|
|
} catch(ErrnoException e) {
|
|
if(e.errno == EPIPE) exit(128+13,true);
|
|
throw e;
|
|
}
|
|
}
|
|
|
|
private int sys_recvfrom(int fdn, int addr, int count, int flags, int sourceAddr, int socklenAddr) throws ErrnoException, FaultException {
|
|
SocketFD fd = getSocketFD(fdn);
|
|
if(flags != 0) throw new ErrnoException(EINVAL);
|
|
|
|
InetAddress[] inetAddr = sourceAddr == 0 ? null : new InetAddress[1];
|
|
int[] port = sourceAddr == 0 ? null : new int[1];
|
|
|
|
count = Math.min(count,MAX_CHUNK);
|
|
byte[] buf = byteBuf(count);
|
|
int n = fd.recvfrom(buf,0,count,inetAddr,port);
|
|
copyout(buf,addr,n);
|
|
|
|
if(sourceAddr != 0) {
|
|
memWrite(sourceAddr,(AF_INET << 16) | port[0]);
|
|
byte[] ip = inetAddr[0].getAddress();
|
|
copyout(ip,sourceAddr+4,4);
|
|
}
|
|
|
|
return n;
|
|
}
|
|
|
|
private int sys_select(int n, int readFDs, int writeFDs, int exceptFDs, int timevalAddr) throws ReadFaultException, ErrnoException {
|
|
return -ENOSYS;
|
|
}
|
|
|
|
private static String hostName() {
|
|
try {
|
|
return InetAddress.getLocalHost().getHostName();
|
|
} catch(UnknownHostException e) {
|
|
return "darkstar";
|
|
}
|
|
}
|
|
|
|
private int sys_sysctl(int nameaddr, int namelen, int oldp, int oldlenaddr, int newp, int newlen) throws FaultException {
|
|
if(newp != 0) return -EPERM;
|
|
if(namelen == 0) return -ENOENT;
|
|
if(oldp == 0) return 0;
|
|
|
|
Object o = null;
|
|
switch(memRead(nameaddr)) {
|
|
case CTL_KERN:
|
|
if(namelen != 2) break;
|
|
switch(memRead(nameaddr+4)) {
|
|
case KERN_OSTYPE:
|
|
o = "NestedVM";
|
|
break;
|
|
case KERN_HOSTNAME:
|
|
o = hostName();
|
|
break;
|
|
case KERN_OSRELEASE:
|
|
o = VERSION;
|
|
break;
|
|
case KERN_VERSION:
|
|
o = "NestedVM Kernel Version " + VERSION;
|
|
break;
|
|
}
|
|
break;
|
|
case CTL_HW:
|
|
if(namelen != 2) break;
|
|
switch(memRead(nameaddr+4)) {
|
|
case HW_MACHINE:
|
|
o = "NestedVM Virtual Machine";
|
|
break;
|
|
}
|
|
break;
|
|
}
|
|
if(o == null) return -ENOENT;
|
|
int len = memRead(oldlenaddr);
|
|
if(o instanceof String) {
|
|
byte[] b = getNullTerminatedBytes((String)o);
|
|
if(len < b.length) return -ENOMEM;
|
|
len = b.length;
|
|
copyout(b,oldp,len);
|
|
memWrite(oldlenaddr,len);
|
|
} else if(o instanceof Integer) {
|
|
if(len < 4) return -ENOMEM;
|
|
memWrite(oldp,((Integer)o).intValue());
|
|
} else {
|
|
throw new Error("should never happen");
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
public static final class GlobalState {
|
|
Hashtable<String,GlobalState.CacheEnt> execCache = new Hashtable<String,GlobalState.CacheEnt>();
|
|
|
|
final UnixRuntime[] tasks;
|
|
int nextPID = 1;
|
|
|
|
/** Table of all current file locks held by this process. */
|
|
Seekable.Lock[] locks = new Seekable.Lock[16];
|
|
|
|
private MP[] mps = new MP[0];
|
|
private FS root;
|
|
|
|
public GlobalState() {
|
|
this(255);
|
|
}
|
|
|
|
public GlobalState(int maxProcs) {
|
|
this(maxProcs,true);
|
|
}
|
|
|
|
public GlobalState(int maxProcs, boolean defaultMounts) {
|
|
tasks = new UnixRuntime[maxProcs+1];
|
|
if(defaultMounts) {
|
|
File root = null;
|
|
if(Platform.getProperty("nestedvm.root") != null) {
|
|
root = new File(Platform.getProperty("nestedvm.root"));
|
|
if(!root.isDirectory()) throw new IllegalArgumentException("nestedvm.root is not a directory");
|
|
} else {
|
|
String cwd = Platform.getProperty("user.dir");
|
|
root = Platform.getRoot(new File(cwd != null ? cwd : "."));
|
|
}
|
|
|
|
addMount("/",new HostFS(root));
|
|
|
|
if(Platform.getProperty("nestedvm.root") == null) {
|
|
File[] roots = Platform.listRoots();
|
|
for(int i=0; i<roots.length; i++) {
|
|
String name = roots[i].getPath();
|
|
if(name.endsWith(File.separator))
|
|
name = name.substring(0,name.length()-1);
|
|
if(name.length() == 0 || name.indexOf('/') != -1) continue;
|
|
addMount("/" + name.toLowerCase(),new HostFS(roots[i]));
|
|
}
|
|
}
|
|
|
|
addMount("/dev",new DevFS());
|
|
addMount("/resource",new ResourceFS());
|
|
addMount("/cygdrive",new CygdriveFS());
|
|
}
|
|
}
|
|
|
|
public String mapHostPath(String s) {
|
|
return mapHostPath(new File(s));
|
|
}
|
|
|
|
public String mapHostPath(File f) {
|
|
MP[] list;
|
|
FS root;
|
|
synchronized(this) {
|
|
mps = this.mps;
|
|
root = this.root;
|
|
}
|
|
if(!f.isAbsolute()) f = new File(f.getAbsolutePath());
|
|
for(int i=mps.length; i>=0; i--) {
|
|
FS fs = i == mps.length ? root : mps[i].fs;
|
|
String path = i == mps.length ? "" : mps[i].path;
|
|
if(!(fs instanceof HostFS)) continue;
|
|
File fsroot = ((HostFS)fs).getRoot();
|
|
if(!fsroot.isAbsolute()) fsroot = new File(fsroot.getAbsolutePath());
|
|
if(f.getPath().startsWith(fsroot.getPath())) {
|
|
char sep = File.separatorChar;
|
|
String child = f.getPath().substring(fsroot.getPath().length());
|
|
if(sep != '/') {
|
|
char[] child_ = child.toCharArray();
|
|
for(int j=0; j<child_.length; j++) {
|
|
if(child_[j] == '/') child_[j] = sep;
|
|
else if(child_[j] == sep) child_[j] = '/';
|
|
}
|
|
child = new String(child_);
|
|
}
|
|
String mapped = "/" + (path.length()==0?"":path+"/") + child;
|
|
return mapped;
|
|
}
|
|
}
|
|
return null;
|
|
}
|
|
|
|
static class MP implements Sort.Comparable {
|
|
public MP(String path, FS fs) {
|
|
this.path = path;
|
|
this.fs = fs;
|
|
}
|
|
public String path;
|
|
public FS fs;
|
|
public int compareTo(Object o) {
|
|
if(!(o instanceof MP)) return 1;
|
|
return -path.compareTo(((MP)o).path);
|
|
}
|
|
}
|
|
|
|
public synchronized FS getMount(String path) {
|
|
if(!path.startsWith("/")) throw new IllegalArgumentException("Mount point doesn't start with a /");
|
|
if(path.equals("/")) return root;
|
|
path = path.substring(1);
|
|
for(int i=0; i<mps.length; i++)
|
|
if(mps[i].path.equals(path)) return mps[i].fs;
|
|
return null;
|
|
}
|
|
|
|
public synchronized void addMount(String path, FS fs) {
|
|
if(getMount(path) != null) throw new IllegalArgumentException("mount point already exists");
|
|
if(!path.startsWith("/")) throw new IllegalArgumentException("Mount point doesn't start with a /");
|
|
|
|
if(fs.owner != null) fs.owner.removeMount(fs);
|
|
fs.owner = this;
|
|
|
|
if(path.equals("/")) {
|
|
root = fs;
|
|
fs.devno = 1;
|
|
return;
|
|
}
|
|
path = path.substring(1);
|
|
int oldLength = mps.length;
|
|
MP[] newMPS = new MP[oldLength + 1];
|
|
if(oldLength != 0) System.arraycopy(mps,0,newMPS,0,oldLength);
|
|
newMPS[oldLength] = new MP(path,fs);
|
|
Sort.sort(newMPS);
|
|
mps = newMPS;
|
|
int highdevno = 0;
|
|
for(int i=0; i<mps.length; i++) highdevno = max(highdevno,mps[i].fs.devno);
|
|
fs.devno = highdevno + 2;
|
|
}
|
|
|
|
public synchronized void removeMount(FS fs) {
|
|
for(int i=0; i<mps.length; i++) if(mps[i].fs == fs) {
|
|
removeMount(i);
|
|
return;
|
|
}
|
|
throw new IllegalArgumentException("mount point doesn't exist");
|
|
}
|
|
|
|
public synchronized void removeMount(String path) {
|
|
if(!path.startsWith("/")) throw new IllegalArgumentException("Mount point doesn't start with a /");
|
|
if(path.equals("/")) {
|
|
removeMount(-1);
|
|
} else {
|
|
path = path.substring(1);
|
|
int p;
|
|
for(p=0; p<mps.length; p++) if(mps[p].path.equals(path)) break;
|
|
if(p == mps.length) throw new IllegalArgumentException("mount point doesn't exist");
|
|
removeMount(p);
|
|
}
|
|
}
|
|
|
|
private void removeMount(int index) {
|
|
if(index == -1) {
|
|
root.owner = null;
|
|
root = null;
|
|
return;
|
|
}
|
|
MP[] newMPS = new MP[mps.length - 1];
|
|
System.arraycopy(mps,0,newMPS,0,index);
|
|
System.arraycopy(mps,0,newMPS,index,mps.length-index-1);
|
|
mps = newMPS;
|
|
}
|
|
|
|
private static class FSPathPair {
|
|
private final FS filesystem;
|
|
private final String path;
|
|
public FSPathPair(FS filesystem, String path) {
|
|
this.filesystem = filesystem;
|
|
this.path = path;
|
|
}
|
|
|
|
public String getPath() {
|
|
return path;
|
|
}
|
|
|
|
public FS getFS() {
|
|
return filesystem;
|
|
}
|
|
}
|
|
|
|
private FSPathPair findFS(String normalizedPath) {
|
|
int pathLength = normalizedPath.length();
|
|
if (pathLength != 0) {
|
|
MP[] list;
|
|
synchronized(this) {
|
|
list = mps;
|
|
}
|
|
for(int i = 0; i < list.length; i++) {
|
|
MP mp = list[i];
|
|
int mpl = mp.path.length();
|
|
if (normalizedPath.startsWith(mp.path) && (pathLength == mpl || normalizedPath.charAt(mpl) == '/')) {
|
|
return new FSPathPair(mp.fs, pathLength == mpl ? "" : normalizedPath.substring(mpl+1));
|
|
}
|
|
}
|
|
}
|
|
return new FSPathPair(root, normalizedPath);
|
|
|
|
}
|
|
|
|
private Object fsop(int op, UnixRuntime r, String normalizedPath1, String normalizedPath2, int arg1, int arg2) throws ErrnoException {
|
|
// We need to dispatch to the right filesystem.
|
|
switch(op) {
|
|
case FS.LINK:
|
|
// Both paths must be in the same filesystem
|
|
FSPathPair lookup1 = findFS(normalizedPath1), lookup2 = findFS(normalizedPath2);
|
|
FS fs1 = lookup1.getFS(), fs2 = lookup2.getFS();
|
|
if (fs1 != fs2) {
|
|
throw new ErrnoException(EXDEV);
|
|
}
|
|
return fs1.dispatch(op,r,lookup1.getPath(), lookup2.getPath(), arg1, arg2);
|
|
default:
|
|
throw new IllegalArgumentException("Unsupported operation: " + Integer.toString(op));
|
|
}
|
|
}
|
|
|
|
private Object fsop(int op, UnixRuntime r, String normalizedPath, int arg1, int arg2) throws ErrnoException {
|
|
int pl = normalizedPath.length();
|
|
if(pl != 0) {
|
|
MP[] list;
|
|
synchronized(this) {
|
|
list = mps;
|
|
}
|
|
for(int i=0; i<list.length; i++) {
|
|
MP mp = list[i];
|
|
int mpl = mp.path.length();
|
|
if(normalizedPath.startsWith(mp.path) && (pl == mpl || normalizedPath.charAt(mpl) == '/'))
|
|
return mp.fs.dispatch(op,r,pl == mpl ? "" : normalizedPath.substring(mpl+1),arg1,arg2);
|
|
}
|
|
}
|
|
return root.dispatch(op,r,normalizedPath,arg1,arg2);
|
|
}
|
|
|
|
public final FD open(UnixRuntime r, String path, int flags, int mode) throws ErrnoException {
|
|
return (FD) fsop(FS.OPEN,r,path,flags,mode);
|
|
}
|
|
public final FStat stat(UnixRuntime r, String path) throws ErrnoException {
|
|
return (FStat) fsop(FS.STAT,r,path,0,0);
|
|
}
|
|
public final FStat lstat(UnixRuntime r, String path) throws ErrnoException {
|
|
return (FStat) fsop(FS.LSTAT,r,path,0,0);
|
|
}
|
|
public final void mkdir(UnixRuntime r, String path, int mode) throws ErrnoException {
|
|
fsop(FS.MKDIR,r,path,mode,0);
|
|
}
|
|
public final void rmdir(UnixRuntime r, String path) throws ErrnoException {
|
|
fsop(FS.RMDIR,r,path,0,0);
|
|
}
|
|
public final void unlink(UnixRuntime r, String path) throws ErrnoException {
|
|
fsop(FS.UNLINK,r,path,0,0);
|
|
}
|
|
public final void link(UnixRuntime r, String oldpath, String newpath) throws ErrnoException {
|
|
fsop(FS.LINK,r,oldpath,newpath, 0, 0);
|
|
}
|
|
|
|
private static class CacheEnt {
|
|
public final long time;
|
|
public final long size;
|
|
public final Object o;
|
|
public CacheEnt(long time, long size, Object o) {
|
|
this.time = time;
|
|
this.size = size;
|
|
this.o = o;
|
|
}
|
|
}
|
|
}
|
|
|
|
public abstract static class FS {
|
|
static final int OPEN = 1;
|
|
static final int STAT = 2;
|
|
static final int LSTAT = 3;
|
|
static final int MKDIR = 4;
|
|
static final int UNLINK = 5;
|
|
static final int RMDIR = 6;
|
|
static final int LINK = 7;
|
|
|
|
GlobalState owner;
|
|
int devno;
|
|
|
|
Object dispatch(int op, UnixRuntime r, String path, int arg1, int arg2) throws ErrnoException {
|
|
switch(op) {
|
|
case OPEN:
|
|
return open(r,path,arg1,arg2);
|
|
case STAT:
|
|
return stat(r,path);
|
|
case LSTAT:
|
|
return lstat(r,path);
|
|
case MKDIR:
|
|
mkdir(r,path,arg1);
|
|
return null;
|
|
case UNLINK:
|
|
unlink(r,path);
|
|
return null;
|
|
case RMDIR:
|
|
rmdir(r,path);
|
|
return null;
|
|
default:
|
|
throw new IllegalArgumentException("Invalid FSOP: " + Integer.toString(op));
|
|
}
|
|
}
|
|
|
|
Object dispatch(int op, UnixRuntime r, String path1, String path2, int arg1, int arg2) throws ErrnoException {
|
|
switch(op) {
|
|
case LINK:
|
|
link(r, path1, path2);
|
|
return null;
|
|
default:
|
|
throw new IllegalArgumentException("Invalid FSOP: " + Integer.toString(op));
|
|
}
|
|
}
|
|
|
|
public FStat lstat(UnixRuntime r, String path) throws ErrnoException {
|
|
return stat(r,path);
|
|
}
|
|
|
|
// If this returns null it'll be truned into an ENOENT
|
|
public abstract FD open(UnixRuntime r, String path, int flags, int mode) throws ErrnoException;
|
|
// If this returns null it'll be turned into an ENOENT
|
|
public abstract FStat stat(UnixRuntime r, String path) throws ErrnoException;
|
|
public abstract void mkdir(UnixRuntime r, String path, int mode) throws ErrnoException;
|
|
public abstract void unlink(UnixRuntime r, String path) throws ErrnoException;
|
|
public abstract void rmdir(UnixRuntime r, String path) throws ErrnoException;
|
|
public abstract void link(UnixRuntime r, String oldpath, String newpath) throws ErrnoException;
|
|
}
|
|
|
|
// chroot support should go in here if it is ever implemented
|
|
private String normalizePath(String path) {
|
|
boolean absolute = path.startsWith("/");
|
|
int cwdl = cwd.length();
|
|
|
|
// NOTE: This isn't just a fast path, it handles cases the code below doesn't
|
|
if(!path.startsWith(".") && path.indexOf("./") == -1 && path.indexOf("//") == -1 && !path.endsWith("."))
|
|
return absolute ? path.substring(1) : cwdl == 0 ? path : path.length() == 0 ? cwd : cwd + "/" + path;
|
|
|
|
char[] in = new char[path.length()+1];
|
|
char[] out = new char[in.length + (absolute ? -1 : cwd.length())];
|
|
path.getChars(0,path.length(),in,0);
|
|
int inp=0, outp=0;
|
|
|
|
if(absolute) {
|
|
do {
|
|
inp++;
|
|
}
|
|
while(in[inp] == '/');
|
|
} else if(cwdl != 0) {
|
|
cwd.getChars(0,cwdl,out,0);
|
|
outp = cwdl;
|
|
}
|
|
|
|
while(in[inp] != 0) {
|
|
if(inp != 0) {
|
|
while(in[inp] != 0 && in[inp] != '/') {
|
|
out[outp++] = in[inp++];
|
|
}
|
|
if(in[inp] == '\0') break;
|
|
while(in[inp] == '/') inp++;
|
|
}
|
|
|
|
// Just read a /
|
|
if(in[inp] == '\0') break;
|
|
if(in[inp] != '.') {
|
|
out[outp++] = '/';
|
|
out[outp++] = in[inp++];
|
|
continue;
|
|
}
|
|
// Just read a /.
|
|
if(in[inp+1] == '\0' || in[inp+1] == '/') {
|
|
inp++;
|
|
continue;
|
|
}
|
|
if(in[inp+1] == '.' && (in[inp+2] == '\0' || in[inp+2] == '/')) { // ..
|
|
// Just read a /..{$,/}
|
|
inp += 2;
|
|
if(outp > 0) outp--;
|
|
while(outp > 0 && out[outp] != '/') outp--;
|
|
//System.err.println("After ..: " + new String(out,0,outp));
|
|
continue;
|
|
}
|
|
// Just read a /.[^.] or /..[^/$]
|
|
inp++;
|
|
out[outp++] = '/';
|
|
out[outp++] = '.';
|
|
}
|
|
if(outp > 0 && out[outp-1] == '/') outp--;
|
|
//System.err.println("normalize: " + path + " -> " + new String(out,0,outp) + " (cwd: " + cwd + ")");
|
|
int outStart = out[0] == '/' ? 1 : 0;
|
|
return new String(out,outStart,outp - outStart);
|
|
}
|
|
|
|
FStat hostFStat(final File f, Object data) {
|
|
boolean e = false;
|
|
try {
|
|
FileInputStream fis = new FileInputStream(f);
|
|
switch(fis.read()) {
|
|
case '\177':
|
|
e = fis.read() == 'E' && fis.read() == 'L' && fis.read() == 'F';
|
|
break;
|
|
case '#':
|
|
e = fis.read() == '!';
|
|
}
|
|
fis.close();
|
|
} catch(IOException e2) { }
|
|
HostFS fs = (HostFS) data;
|
|
final int inode = fs.inodes.get(f.getAbsolutePath());
|
|
final int devno = fs.devno;
|
|
return new HostFStat(f,e) {
|
|
public int inode() {
|
|
return inode;
|
|
}
|
|
public int dev() {
|
|
return devno;
|
|
}
|
|
};
|
|
}
|
|
|
|
FD hostFSDirFD(File f, Object _fs) {
|
|
HostFS fs = (HostFS) _fs;
|
|
return fs.new HostDirFD(f);
|
|
}
|
|
|
|
public static class HostFS extends FS {
|
|
InodeCache inodes = new InodeCache(4000);
|
|
protected File root;
|
|
public File getRoot() {
|
|
return root;
|
|
}
|
|
|
|
protected File hostFile(String path) {
|
|
char sep = File.separatorChar;
|
|
if(sep != '/') {
|
|
char buf[] = path.toCharArray();
|
|
for(int i=0; i<buf.length; i++) {
|
|
char c = buf[i];
|
|
if(c == '/') buf[i] = sep;
|
|
else if(c == sep) buf[i] = '/';
|
|
}
|
|
path = new String(buf);
|
|
}
|
|
return new File(root,path);
|
|
}
|
|
|
|
public HostFS(String root) {
|
|
this(new File(root));
|
|
}
|
|
public HostFS(File root) {
|
|
this.root = root;
|
|
}
|
|
|
|
public FD open(UnixRuntime r, String path, int flags, int mode) throws ErrnoException {
|
|
final File f = hostFile(path);
|
|
return r.hostFSOpen(f,flags,mode,this);
|
|
}
|
|
|
|
public void rmdir(UnixRuntime r, String path) throws ErrnoException {
|
|
unlink(r, path);
|
|
}
|
|
|
|
public void unlink(UnixRuntime r, String path) throws ErrnoException {
|
|
File f = hostFile(path);
|
|
if(r.sm != null && !r.sm.allowUnlink(f)) throw new ErrnoException(EPERM);
|
|
if(!f.exists()) throw new ErrnoException(ENOENT);
|
|
if(!f.delete()) {
|
|
// Can't delete file immediately, so mark for
|
|
// delete on close all matching FDs
|
|
boolean marked = false;
|
|
for(int i=0; i<OPEN_MAX; i++) {
|
|
if(r.fds[i] != null) {
|
|
String fdpath = r.fds[i].getNormalizedPath();
|
|
if(fdpath != null && fdpath.equals(path)) {
|
|
r.fds[i].markDeleteOnClose();
|
|
marked = true;
|
|
}
|
|
}
|
|
}
|
|
if(!marked) throw new ErrnoException(EPERM);
|
|
}
|
|
}
|
|
|
|
public void link(UnixRuntime r, String oldpath, String newpath) throws ErrnoException {
|
|
try {
|
|
Files.createLink(hostFile(newpath).toPath(), hostFile(oldpath).toPath());
|
|
} catch (IOException ex) {
|
|
throw new ErrnoException(EIO);
|
|
}
|
|
}
|
|
|
|
public FStat stat(UnixRuntime r, String path) throws ErrnoException {
|
|
File f = hostFile(path);
|
|
if(r.sm != null && !r.sm.allowStat(f)) throw new ErrnoException(EACCES);
|
|
if(!f.exists()) return null;
|
|
return r.hostFStat(f,this);
|
|
}
|
|
|
|
public void mkdir(UnixRuntime r, String path, int mode) throws ErrnoException {
|
|
File f = hostFile(path);
|
|
if(r.sm != null && !r.sm.allowWrite(f)) throw new ErrnoException(EACCES);
|
|
if(f.exists() && f.isDirectory()) throw new ErrnoException(EEXIST);
|
|
if(f.exists()) throw new ErrnoException(ENOTDIR);
|
|
File parent = getParentFile(f);
|
|
if(parent!=null && (!parent.exists() || !parent.isDirectory())) throw new ErrnoException(ENOTDIR);
|
|
if(!f.mkdir()) throw new ErrnoException(EIO);
|
|
}
|
|
|
|
private static File getParentFile(File f) {
|
|
String p = f.getParent();
|
|
return p == null ? null : new File(p);
|
|
}
|
|
|
|
public class HostDirFD extends DirFD {
|
|
private final File f;
|
|
private final File[] children;
|
|
public HostDirFD(File f) {
|
|
this.f = f;
|
|
String[] l = f.list();
|
|
children = new File[l.length];
|
|
for(int i=0; i<l.length; i++) children[i] = new File(f,l[i]);
|
|
}
|
|
public int size() {
|
|
return children.length;
|
|
}
|
|
public String name(int n) {
|
|
return children[n].getName();
|
|
}
|
|
public int inode(int n) {
|
|
return inodes.get(children[n].getAbsolutePath());
|
|
}
|
|
public int parentInode() {
|
|
File parent = getParentFile(f);
|
|
// HACK: myInode() isn't really correct if we're not the root
|
|
return parent == null ? myInode() : inodes.get(parent.getAbsolutePath());
|
|
}
|
|
public int myInode() {
|
|
return inodes.get(f.getAbsolutePath());
|
|
}
|
|
public int myDev() {
|
|
return devno;
|
|
}
|
|
}
|
|
}
|
|
|
|
/* Implements the Cygwin notation for accessing MS Windows drive letters
|
|
* in a unix path. The path /cygdrive/c/myfile is converted to C:\file.
|
|
* As there is no POSIX standard for this, little checking is done. */
|
|
public static class CygdriveFS extends HostFS {
|
|
protected File hostFile(String path) {
|
|
java.util.logging.Logger.getLogger(CygdriveFS.class.getName()).fine("path = " + path);
|
|
final char drive = path.charAt(0);
|
|
|
|
if (drive < 'a' || drive > 'z' || (path.length() > 1 && path.charAt(1) != '/'))
|
|
return null;
|
|
if (path.length() == 1) {
|
|
path = path + "/";
|
|
}
|
|
|
|
path = drive + ":" + path.substring(1).replace('/', '\\');
|
|
java.util.logging.Logger.getLogger(CygdriveFS.class.getName()).fine("path new = " + path);
|
|
return new File(path);
|
|
}
|
|
|
|
public CygdriveFS() {
|
|
super("/");
|
|
}
|
|
}
|
|
|
|
private static void putInt(byte[] buf, int off, int n) {
|
|
buf[off+0] = (byte)((n>>>24)&0xff);
|
|
buf[off+1] = (byte)((n>>>16)&0xff);
|
|
buf[off+2] = (byte)((n>>> 8)&0xff);
|
|
buf[off+3] = (byte)((n>>> 0)&0xff);
|
|
}
|
|
|
|
public static abstract class DirFD extends FD {
|
|
private int pos = -2;
|
|
|
|
protected abstract int size();
|
|
protected abstract String name(int n);
|
|
protected abstract int inode(int n);
|
|
protected abstract int myDev();
|
|
protected abstract int parentInode();
|
|
protected abstract int myInode();
|
|
public int flags() {
|
|
return O_RDONLY;
|
|
}
|
|
|
|
public int getdents(byte[] buf, int off, int len) {
|
|
int ooff = off;
|
|
int ino;
|
|
int reclen;
|
|
OUTER: for(; len > 0 && pos < size(); pos++) {
|
|
switch(pos) {
|
|
case -2:
|
|
case -1:
|
|
ino = pos == -1 ? parentInode() : myInode();
|
|
if(ino == -1) continue;
|
|
reclen = 9 + (pos == -1 ? 2 : 1);
|
|
if(reclen > len) break OUTER;
|
|
buf[off+8] = '.';
|
|
if(pos == -1) buf[off+9] = '.';
|
|
break;
|
|
default: {
|
|
String f = name(pos);
|
|
byte[] fb = getBytes(f);
|
|
reclen = fb.length + 9;
|
|
if(reclen > len) break OUTER;
|
|
ino = inode(pos);
|
|
System.arraycopy(fb,0,buf,off+8,fb.length);
|
|
}
|
|
}
|
|
buf[off+reclen-1] = 0; // null terminate
|
|
reclen = (reclen + 3) & ~3; // add padding
|
|
putInt(buf,off,reclen);
|
|
putInt(buf,off+4,ino);
|
|
off += reclen;
|
|
len -= reclen;
|
|
}
|
|
return off-ooff;
|
|
}
|
|
|
|
protected FStat _fstat() {
|
|
return new FStat() {
|
|
public int type() {
|
|
return S_IFDIR;
|
|
}
|
|
public int inode() {
|
|
return myInode();
|
|
}
|
|
public int dev() {
|
|
return myDev();
|
|
}
|
|
};
|
|
}
|
|
}
|
|
|
|
public static class DevFS extends FS {
|
|
private static final int ROOT_INODE = 1;
|
|
private static final int NULL_INODE = 2;
|
|
private static final int ZERO_INODE = 3;
|
|
private static final int FD_INODE = 4;
|
|
private static final int FD_INODES = 32;
|
|
|
|
private abstract class DevFStat extends FStat {
|
|
public int dev() {
|
|
return devno;
|
|
}
|
|
public int mode() {
|
|
return 0666;
|
|
}
|
|
public int type() {
|
|
return S_IFCHR;
|
|
}
|
|
public int nlink() {
|
|
return 1;
|
|
}
|
|
public abstract int inode();
|
|
}
|
|
|
|
private abstract class DevDirFD extends DirFD {
|
|
public int myDev() {
|
|
return devno;
|
|
}
|
|
}
|
|
|
|
private FD devZeroFD = new FD() {
|
|
public int read(byte[] a, int off, int length) {
|
|
/*Arrays.fill(a,off,off+length,(byte)0);*/
|
|
for(int i=off; i<off+length; i++) a[i] = 0;
|
|
return length;
|
|
}
|
|
public int write(byte[] a, int off, int length) {
|
|
return length;
|
|
}
|
|
public int seek(int n, int whence) {
|
|
return 0;
|
|
}
|
|
public FStat _fstat() {
|
|
return new DevFStat() {
|
|
public int inode() {
|
|
return ZERO_INODE;
|
|
}
|
|
};
|
|
}
|
|
public int flags() {
|
|
return O_RDWR;
|
|
}
|
|
};
|
|
private FD devNullFD = new FD() {
|
|
public int read(byte[] a, int off, int length) {
|
|
return 0;
|
|
}
|
|
public int write(byte[] a, int off, int length) {
|
|
return length;
|
|
}
|
|
public int seek(int n, int whence) {
|
|
return 0;
|
|
}
|
|
public FStat _fstat() {
|
|
return new DevFStat() {
|
|
public int inode() {
|
|
return NULL_INODE;
|
|
}
|
|
};
|
|
}
|
|
public int flags() {
|
|
return O_RDWR;
|
|
}
|
|
};
|
|
|
|
public FD open(UnixRuntime r, String path, int mode, int flags) throws ErrnoException {
|
|
if(path.equals("null")) return devNullFD;
|
|
if(path.equals("zero")) return devZeroFD;
|
|
if(path.startsWith("fd/")) {
|
|
int n;
|
|
try {
|
|
n = Integer.parseInt(path.substring(3));
|
|
} catch(NumberFormatException e) {
|
|
return null;
|
|
}
|
|
if(n < 0 || n >= OPEN_MAX) return null;
|
|
if(r.fds[n] == null) return null;
|
|
return r.fds[n].dup();
|
|
}
|
|
if(path.equals("fd")) {
|
|
int count=0;
|
|
for(int i=0; i<OPEN_MAX; i++) if(r.fds[i] != null) {
|
|
count++;
|
|
}
|
|
final int[] files = new int[count];
|
|
count = 0;
|
|
for(int i=0; i<OPEN_MAX; i++) if(r.fds[i] != null) files[count++] = i;
|
|
return new DevDirFD() {
|
|
public int myInode() {
|
|
return FD_INODE;
|
|
}
|
|
public int parentInode() {
|
|
return ROOT_INODE;
|
|
}
|
|
public int inode(int n) {
|
|
return FD_INODES + n;
|
|
}
|
|
public String name(int n) {
|
|
return Integer.toString(files[n]);
|
|
}
|
|
public int size() {
|
|
return files.length;
|
|
}
|
|
};
|
|
}
|
|
if(path.equals("")) {
|
|
return new DevDirFD() {
|
|
public int myInode() {
|
|
return ROOT_INODE;
|
|
}
|
|
// HACK: We don't have any clean way to get the parent inode
|
|
public int parentInode() {
|
|
return ROOT_INODE;
|
|
}
|
|
public int inode(int n) {
|
|
switch(n) {
|
|
case 0:
|
|
return NULL_INODE;
|
|
case 1:
|
|
return ZERO_INODE;
|
|
case 2:
|
|
return FD_INODE;
|
|
default:
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
public String name(int n) {
|
|
switch(n) {
|
|
case 0:
|
|
return "null";
|
|
case 1:
|
|
return "zero";
|
|
case 2:
|
|
return "fd";
|
|
default:
|
|
return null;
|
|
}
|
|
}
|
|
public int size() {
|
|
return 3;
|
|
}
|
|
};
|
|
}
|
|
return null;
|
|
}
|
|
|
|
public FStat stat(UnixRuntime r,String path) throws ErrnoException {
|
|
if(path.equals("null")) return devNullFD.fstat();
|
|
if(path.equals("zero")) return devZeroFD.fstat();
|
|
if(path.startsWith("fd/")) {
|
|
int n;
|
|
try {
|
|
n = Integer.parseInt(path.substring(3));
|
|
} catch(NumberFormatException e) {
|
|
return null;
|
|
}
|
|
if(n < 0 || n >= OPEN_MAX) return null;
|
|
if(r.fds[n] == null) return null;
|
|
return r.fds[n].fstat();
|
|
}
|
|
if(path.equals("fd")) return new FStat() {
|
|
public int inode() {
|
|
return FD_INODE;
|
|
}
|
|
public int dev() {
|
|
return devno;
|
|
}
|
|
public int type() {
|
|
return S_IFDIR;
|
|
}
|
|
public int mode() {
|
|
return 0444;
|
|
}
|
|
};
|
|
if(path.equals("")) return new FStat() {
|
|
public int inode() {
|
|
return ROOT_INODE;
|
|
}
|
|
public int dev() {
|
|
return devno;
|
|
}
|
|
public int type() {
|
|
return S_IFDIR;
|
|
}
|
|
public int mode() {
|
|
return 0444;
|
|
}
|
|
};
|
|
return null;
|
|
}
|
|
|
|
public void mkdir(UnixRuntime r, String path, int mode) throws ErrnoException {
|
|
throw new ErrnoException(EROFS);
|
|
}
|
|
public void rmdir(UnixRuntime r, String path) throws ErrnoException {
|
|
throw new ErrnoException(EROFS);
|
|
}
|
|
public void unlink(UnixRuntime r, String path) throws ErrnoException {
|
|
throw new ErrnoException(EROFS);
|
|
}
|
|
public void link(UnixRuntime r, String oldpath, String newpath) throws ErrnoException {
|
|
throw new ErrnoException(EROFS);
|
|
}
|
|
}
|
|
|
|
|
|
public static class ResourceFS extends FS {
|
|
final InodeCache inodes = new InodeCache(500);
|
|
|
|
public FStat lstat(UnixRuntime r, String path) throws ErrnoException {
|
|
return stat(r,path);
|
|
}
|
|
public void mkdir(UnixRuntime r, String path, int mode) throws ErrnoException {
|
|
throw new ErrnoException(EROFS);
|
|
}
|
|
public void unlink(UnixRuntime r, String path) throws ErrnoException {
|
|
throw new ErrnoException(EROFS);
|
|
}
|
|
public void rmdir(UnixRuntime r, String path) throws ErrnoException {
|
|
throw new ErrnoException(EROFS);
|
|
}
|
|
public void link(UnixRuntime r, String oldpath, String newpath) throws ErrnoException {
|
|
throw new ErrnoException(EROFS);
|
|
}
|
|
|
|
FStat connFStat(final URLConnection conn) {
|
|
return new FStat() {
|
|
public int type() {
|
|
return S_IFREG;
|
|
}
|
|
public int nlink() {
|
|
return 1;
|
|
}
|
|
public int mode() {
|
|
return 0444;
|
|
}
|
|
public int size() {
|
|
return conn.getContentLength();
|
|
}
|
|
public int mtime() {
|
|
return (int)(conn.getDate() / 1000);
|
|
}
|
|
public int inode() {
|
|
return inodes.get(conn.getURL().toString());
|
|
}
|
|
public int dev() {
|
|
return devno;
|
|
}
|
|
};
|
|
}
|
|
|
|
public FStat stat(UnixRuntime r, String path) throws ErrnoException {
|
|
URL url = r.getClass().getResource("/" + path);
|
|
if(url == null) return null;
|
|
try {
|
|
return connFStat(url.openConnection());
|
|
} catch(IOException e) {
|
|
throw new ErrnoException(EIO);
|
|
}
|
|
}
|
|
|
|
public FD open(UnixRuntime r, String path, int flags, int mode) throws ErrnoException {
|
|
if((flags & ~3) != 0) {
|
|
if(STDERR_DIAG)
|
|
System.err.println("WARNING: Unsupported flags passed to ResourceFS.open(\"" + path + "\"): " + toHex(flags & ~3));
|
|
throw new ErrnoException(ENOTSUP);
|
|
}
|
|
if((flags&3) != RD_ONLY) throw new ErrnoException(EROFS);
|
|
URL url = r.getClass().getResource("/" + path);
|
|
if(url == null) return null;
|
|
try {
|
|
final URLConnection conn = url.openConnection();
|
|
Seekable.InputStream si = new Seekable.InputStream(conn.getInputStream());
|
|
return new SeekableFD(si,flags) {
|
|
protected FStat _fstat() {
|
|
return connFStat(conn);
|
|
}
|
|
};
|
|
} catch(FileNotFoundException e) {
|
|
if(e.getMessage() != null && e.getMessage().indexOf("Permission denied") >= 0) throw new ErrnoException(EACCES);
|
|
return null;
|
|
} catch(IOException e) {
|
|
throw new ErrnoException(EIO);
|
|
}
|
|
}
|
|
}
|
|
}
|