Skip to content
Snippets Groups Projects
Commit 10a5c1c9 authored by Mick Jordan's avatar Mick Jordan
Browse files

Add support for getting class path to mx_fastr/Rembedded.c

parent 8d91602b
No related branches found
No related tags found
No related merge requests found
...@@ -35,6 +35,7 @@ SA_TYPE SaveAction; // ?? ...@@ -35,6 +35,7 @@ SA_TYPE SaveAction; // ??
typedef jint (JNICALL *JNI_CreateJavaVMFunc) typedef jint (JNICALL *JNI_CreateJavaVMFunc)
(JavaVM **pvm, void **penv, void *args); (JavaVM **pvm, void **penv, void *args);
static void *dlopen_jvmlib(char *libpath) { static void *dlopen_jvmlib(char *libpath) {
void *handle = dlopen(libpath, RTLD_GLOBAL | RTLD_NOW); void *handle = dlopen(libpath, RTLD_GLOBAL | RTLD_NOW);
if (handle == NULL) { if (handle == NULL) {
...@@ -59,11 +60,18 @@ static int process_vmargs(int argc, char *argv[], char *vmargv[], char *uargv[]) ...@@ -59,11 +60,18 @@ static int process_vmargs(int argc, char *argv[], char *vmargv[], char *uargv[])
return vcount; return vcount;
} }
char *get_classpath(char *r_home);
int Rf_initialize_R(int argc, char *argv[]) { int Rf_initialize_R(int argc, char *argv[]) {
if (initialized) { if (initialized) {
fprintf(stderr, "%s", "R is already initialized\n"); fprintf(stderr, "%s", "R is already initialized\n");
exit(1); exit(1);
} }
char *r_home = getenv("R_HOME");
if (r_home == NULL) {
printf("R_HOME must be set\n");
exit(1);
}
struct utsname utsname; struct utsname utsname;
uname(&utsname); uname(&utsname);
char jvmlib_path[256]; char jvmlib_path[256];
...@@ -94,17 +102,8 @@ int Rf_initialize_R(int argc, char *argv[]) { ...@@ -94,17 +102,8 @@ int Rf_initialize_R(int argc, char *argv[]) {
exit(1); exit(1);
} }
// TODO getting the correct classpath is hard, need a helper program char *vm_cp = get_classpath(r_home);
char *vm_cp = getenv("FASTR_CLASSPATH"); //printf("cp %s\n", vm_cp);
if (vm_cp == NULL) {
printf("Rf_initialize_R: FASTR_CLASSPATH env var not set\n");
exit(1);
}
int cplen = (int) strlen(vm_cp);
char *cp = malloc(cplen + 32);
strcpy(cp, "-Djava.class.path=");
strcat(cp, vm_cp);
char **vmargs = malloc(argc * sizeof(char*)); char **vmargs = malloc(argc * sizeof(char*));
char **uargs = malloc(argc * sizeof(char*)); char **uargs = malloc(argc * sizeof(char*));
...@@ -113,7 +112,7 @@ int Rf_initialize_R(int argc, char *argv[]) { ...@@ -113,7 +112,7 @@ int Rf_initialize_R(int argc, char *argv[]) {
argv = uargs; argv = uargs;
JavaVMOption vm_options[1 + vmargc]; JavaVMOption vm_options[1 + vmargc];
vm_options[0].optionString = cp; vm_options[0].optionString = vm_cp;
for (int i = 0; i < vmargc; i++) { for (int i = 0; i < vmargc; i++) {
vm_options[i + 1].optionString = vmargs[i]; vm_options[i + 1].optionString = vmargs[i];
} }
...@@ -432,3 +431,70 @@ void R_runHandlers(InputHandler *handlers, fd_set *mask) { ...@@ -432,3 +431,70 @@ void R_runHandlers(InputHandler *handlers, fd_set *mask) {
} }
int R_wait_usec; int R_wait_usec;
#include <unistd.h>
#include <errno.h>
void perror_exit(char *msg) {
perror(msg);
exit(1);
}
// support for getting the correct classpath for the VM
// We use $R_HOME/bin/exec/Rclasspath to do this to emulate what happens
// during normal execution
char *get_classpath(char *r_home) {
int pipefd[2];
if (pipe(pipefd) == -1) {
perror_exit("pipe");
}
pid_t pid = fork();
if (pid == -1) {
perror("fork");
}
if (pid == 0) {
// child
char path[1024];
strcpy(path, r_home);
strcat(path, "/bin/exec/Rclasspath");
while ((dup2(pipefd[1], STDOUT_FILENO) == -1) && (errno == EINTR)) {}
close(pipefd[1]);
close(pipefd[0]);
int rc = execl(path, (char *)NULL);
if (rc == -1) {
perror_exit("exec");
}
return NULL;
} else {
// parent
const char *cpdef = "-Djava.class.path=";
char *buf = malloc(4096);
strcpy(buf, cpdef);
char *bufptr = buf + strlen(cpdef);
int max = 4096 - strlen(cpdef);
close(pipefd[1]);
while (1) {
int count = read(pipefd[0], bufptr, max);
if (count == -1) {
if (errno == EINTR) {
continue;
} else {
perror_exit("read");
}
} else if (count == 0) {
// scrub any newline
bufptr--;
if (*bufptr != '\n') {
bufptr++;
}
*bufptr = 0;
break;
} else {
bufptr += count;
}
}
close(pipefd[0]);
wait(NULL);
return buf;
}
}
...@@ -65,7 +65,8 @@ $(FASTR_BIN_DIR)/R: Makefile R.sh Rscript.sh Rscript_exec.sh ...@@ -65,7 +65,8 @@ $(FASTR_BIN_DIR)/R: Makefile R.sh Rscript.sh Rscript_exec.sh
cp R.sh $(FASTR_BIN_DIR)/exec/R cp R.sh $(FASTR_BIN_DIR)/exec/R
cp Rscript_exec.sh $(FASTR_BIN_DIR)/execRscript/Rscript cp Rscript_exec.sh $(FASTR_BIN_DIR)/execRscript/Rscript
cp Rscript.sh $(FASTR_BIN_DIR)/Rscript cp Rscript.sh $(FASTR_BIN_DIR)/Rscript
chmod +x $(FASTR_BIN_DIR)/exec/R $(FASTR_BIN_DIR)/execRscript/Rscript $(FASTR_BIN_DIR)/Rscript cp Rclasspath.sh $(FASTR_BIN_DIR)/exec/Rclasspath
chmod +x $(FASTR_BIN_DIR)/exec/R $(FASTR_BIN_DIR)/execRscript/Rscript $(FASTR_BIN_DIR)/Rscript $(FASTR_BIN_DIR)/exec/Rclasspath
cp $(SUPPORT_SCRIPTS) $(FASTR_BIN_DIR) cp $(SUPPORT_SCRIPTS) $(FASTR_BIN_DIR)
sed -e 's!^\(R_HOME_DIR=\)\(.*\)!\1"$(FASTR_R_HOME)"!' < $(R_SCRIPT) > $(FASTR_BIN_DIR)/R sed -e 's!^\(R_HOME_DIR=\)\(.*\)!\1"$(FASTR_R_HOME)"!' < $(R_SCRIPT) > $(FASTR_BIN_DIR)/R
chmod +x $(FASTR_BIN_DIR)/R chmod +x $(FASTR_BIN_DIR)/R
......
...@@ -117,6 +117,9 @@ def do_run_r(args, command, extraVmArgs=None, jdk=None, **kwargs): ...@@ -117,6 +117,9 @@ def do_run_r(args, command, extraVmArgs=None, jdk=None, **kwargs):
vmArgs.append(_command_class_dict[command.lower()]) vmArgs.append(_command_class_dict[command.lower()])
return mx.run_java(vmArgs + args, jdk=jdk, **kwargs) return mx.run_java(vmArgs + args, jdk=jdk, **kwargs)
def r_classpath(args):
print mx.classpath(_r_command_project)
def _sanitize_vmArgs(jdk, vmArgs): def _sanitize_vmArgs(jdk, vmArgs):
''' '''
jdk/vm dependent analysis of vmArgs to remove those that are not appropriate for the jdk/vm dependent analysis of vmArgs to remove those that are not appropriate for the
...@@ -535,6 +538,7 @@ _commands = { ...@@ -535,6 +538,7 @@ _commands = {
'rembed' : [rembed, '[options]'], 'rembed' : [rembed, '[options]'],
'installpkgs' : [installpkgs, '[options]'], 'installpkgs' : [installpkgs, '[options]'],
'installcran' : [installpkgs, '[options]'], 'installcran' : [installpkgs, '[options]'],
'r-cp' : [r_classpath, '[options]'],
} }
mx.update_commands(_fastr_suite, _commands) mx.update_commands(_fastr_suite, _commands)
0% Loading or .
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment