我正在尝试使用JNI将C程序移植到android。我已经能够设置程序并使java和c一起正常工作。问题是我需要能够使用STDIN,因为C程序从STDIN读取输入并通过STDOUT返回响应(C程序是服务器 - 客户端应用程序)。 我不知道是否值得一提,但C程序使用STDIN_FILENO文件描述符来读取STDIN的输入。 如何使用Java从STDOUT和WRITE读取到STDIN?
我做了一些研究,并在以下链接中找到了一些模糊的解释:https://groups.google.com/forum/#!topic/android-ndk/Brm6jPr4C0Y我不明白。
这里是C代码https://github.com/unekwu/kemi/blob/master/jni/dnscat/dnscat.c#L1270
更多详情
C程序通常从命令行运行
dnscat --dns <ip> <port>
之后,它开始侦听来自用户的消息。通常从标准输入。
现在在我的Android应用中, 我可以通过调用main和不同的名称并将ann字符串数组传递给JNI来运行它。我验证程序启动纠正。问题是我将如何将消息发送到程序,因为android上没有stdin。
答案 0 :(得分:7)
我在github上创建了一个项目,您可以从here.下载
它创建了2个命名管道(FIFO),一个用于输入,另一个用于输出。
它在本机代码中以只写模式打开管道的一端,在Java代码中以只读模式打开管道的另一端。本机代码中的文件描述符映射到STDOUT,即1,此后对本机代码中的STDOUT的任何写入都将被重定向到可以用Java代码读取的管道的另一端。
它在本机代码中以只读模式打开管道的一端,在Java代码中以只写模式打开管道的另一端。本机代码中的文件描述符映射到STDIN,即0,此后对Java代码中管道的另一端的任何写入将由本机代码使用STDIN读取。
要实现STDOUT重定向:
原生代码:
/*
* Step 1: Make a named pipe
* Step 2: Open the pipe in Write only mode. Java code will open it in Read only mode.
* Step 3: Make STDOUT i.e. 1, a duplicate of opened pipe file descriptor.
* Step 4: Any writes from now on to STDOUT will be redirected to the the pipe and can be read by Java code.
*/
int out = mkfifo(outfile, 0664);
int fdo = open(outfile, O_WRONLY);
dup2(fdo, 1);
setbuf(stdout, NULL);
fprintf(stdout, "This string will be written to %s", outfile);
fprintf(stdout, "\n");
fflush(stdout);
close(fdo);
Java代码:
/*
* This thread is used for reading content which will be written by native code using STDOUT.
*/
new Thread(new Runnable() {
@Override
public void run() {
BufferedReader in = null;
try {
in = new BufferedReader(new FileReader(mOutfile));
while(in.ready()) {
final String str = in.readLine();
mHandler.post(new Runnable() {
@Override
public void run() {
Toast.makeText(RedirectionJni.this, str, Toast.LENGTH_LONG).show();
}
});
}
in.close();
} catch (FileNotFoundException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
}
}
}).start();
实现STDIN重定向:
原生代码:
/*
* Step 1: Make a named pipe
* Step 2: Open the pipe in Read only mode. Java code will open it in Write only mode.
* Step 3: Make STDIN i.e. 0, a duplicate of opened pipe file descriptor.
* Step 4: Any reads from STDIN, will be actually read from the pipe and JAVA code will perform write operations.
*/
int in = mkfifo(infile, 0664);
int fdi = open(infile, O_RDONLY);
dup2(fdi, 0);
char buf[256] = "";
fscanf(stdin, "%*s %99[^\n]", buf); // Use this format to read white spaces.
close(fdi);
Java代码:
/*
* This thread is used for writing content which will be read by native code using STDIN.
*/
new Thread(new Runnable() {
@Override
public void run() {
BufferedWriter out = null;
try {
out = new BufferedWriter(new OutputStreamWriter(new FileOutputStream(mInfile)));
String content = "This content is written to " + mInfile;
out.write(content.toCharArray(), 0, content.toCharArray().length);
out.flush();
out.close();
} catch (FileNotFoundException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
}
}
}).start();
如果您需要帮助,请告诉我。
答案 1 :(得分:1)
在Java代码中,通过
获取命令行Process p;
p = Runtime.getRuntime().exec("su"); // or execute something else, su is just to get root
DataOutputStream dos = new DataOutputStream(p.getOutputStream());
dos.writeBytes(ForExampleABinaryPath+" &\n");
dos.flush();
// to read (might has to run parallel in different thread)
InputStream is = p.getInputStream();
BufferedReader br = new BufferedReader(new InputStreamReader(is));
is.readLine() // will get you a line o null
使用write&amp;&amp;同时刷新和阅读,您可能能够实现目标。 祝你好运