2014-12-16 21:24:01 +00:00
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <stdio.h>
|
2014-12-16 15:34:10 +00:00
|
|
|
#include <jni.h>
|
|
|
|
#include <sys/types.h>
|
2014-12-16 22:07:26 +00:00
|
|
|
#include <sys/wait.h>
|
2014-12-16 15:34:10 +00:00
|
|
|
#include <signal.h>
|
|
|
|
#include <unistd.h>
|
2014-12-16 21:24:01 +00:00
|
|
|
#include <fcntl.h>
|
|
|
|
|
2014-12-16 22:27:43 +00:00
|
|
|
const char *conf_path = "", *conf_shell = "", *conf_home = "";
|
|
|
|
|
|
|
|
/* NB - this will leak memory like crazy if called often.... */
|
|
|
|
const char *
|
|
|
|
conf_path_file(const char *fn)
|
|
|
|
{
|
|
|
|
char *ret = malloc(strlen(conf_path)+strlen(fn)+20);
|
|
|
|
sprintf(ret, "%s/%s", conf_path, fn);
|
|
|
|
return ret;
|
|
|
|
}
|
2014-12-14 21:20:49 +00:00
|
|
|
|
2014-12-16 15:34:10 +00:00
|
|
|
|
|
|
|
static JNIEnv *env;
|
|
|
|
static jclass cl_string;
|
|
|
|
static jclass cl_simplesshdservice;
|
|
|
|
static jfieldID fid_sss_sshd_pid;
|
|
|
|
|
2014-12-16 21:13:04 +00:00
|
|
|
extern int dropbear_main(int argc, char **argv);
|
|
|
|
|
2014-12-16 15:34:10 +00:00
|
|
|
static int
|
|
|
|
jni_init(JNIEnv *env_)
|
|
|
|
{
|
|
|
|
env = env_;
|
|
|
|
#define CLASS(var, id) \
|
|
|
|
cl_##var = (*env)->FindClass(env, id); \
|
|
|
|
if (!cl_##var) return 0;
|
|
|
|
#define METHOD(var, mycl, id, sig) \
|
|
|
|
mid_##var = (*env)->GetMethodID(env, cl_##mycl, id, sig); \
|
|
|
|
if (!mid_##var) return 0;
|
|
|
|
#define FIELD(var, mycl, id, sig) \
|
|
|
|
fid_##var = (*env)->GetFieldID(env, cl_##mycl, id, sig); \
|
|
|
|
if (!fid_##var) return 0;
|
|
|
|
#define STFIELD(var, mycl, id, sig) \
|
|
|
|
fid_##var = (*env)->GetStaticFieldID(env, cl_##mycl, id, sig); \
|
|
|
|
if (!fid_##var) return 0;
|
|
|
|
|
|
|
|
CLASS(string, "java/lang/String")
|
|
|
|
CLASS(simplesshdservice, "org/galexander/sshd/SimpleSSHDService")
|
|
|
|
|
2014-12-16 16:19:26 +00:00
|
|
|
STFIELD(sss_sshd_pid, simplesshdservice, "sshd_pid", "I")
|
2014-12-16 15:34:10 +00:00
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2014-12-16 20:50:30 +00:00
|
|
|
static const char *
|
|
|
|
from_java_string(jobject s)
|
|
|
|
{
|
|
|
|
const char *ret, *t;
|
|
|
|
t = (*env)->GetStringUTFChars(env, s, NULL);
|
|
|
|
if (!t) {
|
2014-12-16 21:24:01 +00:00
|
|
|
return "";
|
2014-12-16 20:50:30 +00:00
|
|
|
}
|
|
|
|
ret = strdup(t);
|
|
|
|
(*env)->ReleaseStringUTFChars(env, s, t);
|
2014-12-16 21:52:54 +00:00
|
|
|
return ret;
|
2014-12-16 20:50:30 +00:00
|
|
|
}
|
|
|
|
|
2014-12-16 15:34:10 +00:00
|
|
|
JNIEXPORT void JNICALL
|
|
|
|
Java_org_galexander_sshd_SimpleSSHDService_start_1sshd(JNIEnv *env_,
|
2014-12-16 20:50:30 +00:00
|
|
|
jobject this,
|
|
|
|
jint port, jobject jpath, jobject jshell, jobject jhome)
|
2014-12-16 15:34:10 +00:00
|
|
|
{
|
|
|
|
pid_t pid;
|
2014-12-16 20:50:30 +00:00
|
|
|
|
2014-12-16 15:34:10 +00:00
|
|
|
if (!jni_init(env_)) {
|
|
|
|
return;
|
|
|
|
}
|
2014-12-16 21:24:01 +00:00
|
|
|
conf_path = from_java_string(jpath);
|
|
|
|
conf_shell = from_java_string(jshell);
|
|
|
|
conf_home = from_java_string(jhome);
|
2014-12-16 20:50:30 +00:00
|
|
|
|
2014-12-16 15:34:10 +00:00
|
|
|
pid = fork();
|
|
|
|
if (pid == 0) {
|
2014-12-16 22:51:24 +00:00
|
|
|
char *argv[100] = { "sshd", NULL };
|
2014-12-16 21:34:10 +00:00
|
|
|
int argc = 1;
|
2014-12-16 22:27:43 +00:00
|
|
|
const char *logfn;
|
2014-12-16 21:24:01 +00:00
|
|
|
int logfd;
|
2014-12-16 22:14:01 +00:00
|
|
|
int retval;
|
2014-12-16 21:24:01 +00:00
|
|
|
|
2014-12-16 22:27:43 +00:00
|
|
|
logfn = conf_path_file("dropbear.err");
|
2014-12-16 21:24:01 +00:00
|
|
|
unlink(logfn);
|
|
|
|
logfd = open(logfn, O_CREAT|O_WRONLY, 0666);
|
|
|
|
if (logfd != -1) {
|
|
|
|
/* replace stderr, so the output is preserved... */
|
|
|
|
dup2(logfd, 2);
|
|
|
|
}
|
2014-12-16 22:51:24 +00:00
|
|
|
|
|
|
|
argv[argc++] = "-R"; /* enable DROPBEAR_DELAY_HOSTKEY */
|
2014-12-16 23:49:00 +00:00
|
|
|
argv[argc++] = "-F"; /* don't redundant fork to background */
|
2014-12-16 21:34:10 +00:00
|
|
|
if (port) {
|
|
|
|
argv[argc++] = "-p";
|
|
|
|
argv[argc] = malloc(20);
|
2014-12-16 22:56:47 +00:00
|
|
|
sprintf(argv[argc], "0.0.0.0:%d", (int)port);
|
2014-12-16 21:34:10 +00:00
|
|
|
argc++;
|
|
|
|
}
|
2014-12-16 22:14:01 +00:00
|
|
|
fprintf(stderr, "starting dropbear\n");
|
2014-12-16 22:31:14 +00:00
|
|
|
retval = dropbear_main(argc, argv);
|
2014-12-16 22:14:01 +00:00
|
|
|
fprintf(stderr, "dropbear finished (%d)\n", retval);
|
2014-12-16 15:34:10 +00:00
|
|
|
} else {
|
2014-12-16 20:50:30 +00:00
|
|
|
(*env)->SetStaticIntField(env, cl_simplesshdservice,
|
|
|
|
fid_sss_sshd_pid, pid);
|
2014-12-16 15:34:10 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
JNIEXPORT void JNICALL
|
|
|
|
Java_org_galexander_sshd_SimpleSSHDService_stop_1sshd(JNIEnv *env_, jobject this)
|
|
|
|
{
|
|
|
|
pid_t pid;
|
|
|
|
if (!jni_init(env_)) {
|
|
|
|
return;
|
|
|
|
}
|
2014-12-16 16:19:26 +00:00
|
|
|
pid = (*env)->GetStaticIntField(env, cl_simplesshdservice, fid_sss_sshd_pid);
|
2014-12-16 15:34:10 +00:00
|
|
|
kill(pid, SIGKILL);
|
2014-12-16 16:19:26 +00:00
|
|
|
(*env)->SetStaticIntField(env, cl_simplesshdservice, fid_sss_sshd_pid, 0);
|
2014-12-16 15:34:10 +00:00
|
|
|
}
|
2014-12-16 22:07:26 +00:00
|
|
|
|
|
|
|
JNIEXPORT int JNICALL
|
|
|
|
Java_org_galexander_sshd_SimpleSSHDService_waitpid(JNIEnv *env_, jclass cl,
|
|
|
|
jint pid)
|
|
|
|
{
|
|
|
|
int status;
|
|
|
|
waitpid(pid, &status, 0);
|
|
|
|
if (WIFEXITED(status)) {
|
|
|
|
return WEXITSTATUS(status);
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|