Properly solve the connection problem
This commit is contained in:
parent
46447f7cfd
commit
d952cc2327
@ -43,10 +43,15 @@
|
|||||||
<!-- Superuser -->
|
<!-- Superuser -->
|
||||||
<activity
|
<activity
|
||||||
android:name="a.m"
|
android:name="a.m"
|
||||||
android:theme="@android:style/Theme.Translucent.NoTitleBar"
|
|
||||||
android:directBootAware="true"
|
android:directBootAware="true"
|
||||||
android:excludeFromRecents="true"
|
android:excludeFromRecents="true"
|
||||||
android:exported="false" />
|
android:exported="false"
|
||||||
|
tools:ignore="AppLinkUrlError">
|
||||||
|
<intent-filter>
|
||||||
|
<action android:name="android.intent.action.VIEW"/>
|
||||||
|
<category android:name="android.intent.category.DEFAULT"/>
|
||||||
|
</intent-filter>
|
||||||
|
</activity>
|
||||||
|
|
||||||
<!-- Receiver -->
|
<!-- Receiver -->
|
||||||
<receiver
|
<receiver
|
||||||
|
@ -1,5 +1,6 @@
|
|||||||
package com.topjohnwu.magisk.ui.surequest
|
package com.topjohnwu.magisk.ui.surequest
|
||||||
|
|
||||||
|
import android.content.Intent
|
||||||
import android.content.pm.ActivityInfo
|
import android.content.pm.ActivityInfo
|
||||||
import android.os.Build
|
import android.os.Build
|
||||||
import android.os.Bundle
|
import android.os.Bundle
|
||||||
@ -11,6 +12,7 @@ import com.topjohnwu.magisk.model.entity.MagiskPolicy
|
|||||||
import com.topjohnwu.magisk.model.events.DieEvent
|
import com.topjohnwu.magisk.model.events.DieEvent
|
||||||
import com.topjohnwu.magisk.model.events.ViewEvent
|
import com.topjohnwu.magisk.model.events.ViewEvent
|
||||||
import com.topjohnwu.magisk.utils.SuHandler
|
import com.topjohnwu.magisk.utils.SuHandler
|
||||||
|
import com.topjohnwu.magisk.utils.SuHandler.REQUEST
|
||||||
import org.koin.androidx.viewmodel.ext.android.viewModel
|
import org.koin.androidx.viewmodel.ext.android.viewModel
|
||||||
|
|
||||||
open class SuRequestActivity : BaseActivity<SuRequestViewModel, ActivityRequestBinding>() {
|
open class SuRequestActivity : BaseActivity<SuRequestViewModel, ActivityRequestBinding>() {
|
||||||
@ -28,13 +30,28 @@ open class SuRequestActivity : BaseActivity<SuRequestViewModel, ActivityRequestB
|
|||||||
lockOrientation()
|
lockOrientation()
|
||||||
super.onCreate(savedInstanceState)
|
super.onCreate(savedInstanceState)
|
||||||
|
|
||||||
if (intent?.action == SuHandler.REQUEST) {
|
fun showRequest() {
|
||||||
if (!viewModel.handleRequest(intent))
|
if (!viewModel.handleRequest(intent))
|
||||||
finish()
|
finish()
|
||||||
} else {
|
}
|
||||||
SuHandler(this, intent.action, intent.extras)
|
|
||||||
|
fun runHandler(action: String?) {
|
||||||
|
SuHandler(this, action, intent.extras)
|
||||||
finish()
|
finish()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (intent.action == Intent.ACTION_VIEW) {
|
||||||
|
val action = intent.getStringExtra("action")
|
||||||
|
if (action == REQUEST) {
|
||||||
|
showRequest()
|
||||||
|
} else {
|
||||||
|
runHandler(action)
|
||||||
|
}
|
||||||
|
} else if (intent.action == REQUEST) {
|
||||||
|
showRequest()
|
||||||
|
} else {
|
||||||
|
runHandler(intent.action)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
override fun onEventDispatched(event: ViewEvent) {
|
override fun onEventDispatched(event: ViewEvent) {
|
||||||
|
@ -16,7 +16,7 @@ buildscript {
|
|||||||
maven { url 'https://kotlin.bintray.com/kotlinx' }
|
maven { url 'https://kotlin.bintray.com/kotlinx' }
|
||||||
}
|
}
|
||||||
dependencies {
|
dependencies {
|
||||||
classpath 'com.android.tools.build:gradle:3.5.1'
|
classpath 'com.android.tools.build:gradle:3.5.2'
|
||||||
classpath "org.jetbrains.kotlin:kotlin-gradle-plugin:${vKotlin}"
|
classpath "org.jetbrains.kotlin:kotlin-gradle-plugin:${vKotlin}"
|
||||||
|
|
||||||
|
|
||||||
|
@ -13,107 +13,188 @@
|
|||||||
|
|
||||||
using namespace std;
|
using namespace std;
|
||||||
|
|
||||||
|
enum {
|
||||||
|
NAMED_ACTIVITY,
|
||||||
|
PKG_ACTIVITY,
|
||||||
|
CONTENT_PROVIDER
|
||||||
|
};
|
||||||
|
|
||||||
#define CALL_PROVIDER \
|
#define CALL_PROVIDER \
|
||||||
"/system/bin/app_process", "/system/bin", "com.android.commands.content.Content", \
|
"/system/bin/app_process", "/system/bin", "com.android.commands.content.Content", \
|
||||||
"call", "--uri", nullptr, "--user", nullptr, "--method"
|
"call", "--uri", target, "--user", user, "--method", action
|
||||||
|
|
||||||
#define content_exec_info(info, ...) {\
|
#define START_ACTIVITY \
|
||||||
const char *cmd[] = { CALL_PROVIDER, __VA_ARGS__, nullptr }; \
|
"/system/bin/app_process", "/system/bin", "com.android.commands.am.Am", \
|
||||||
exec_content_cmd(cmd, info); \
|
"start", "-p", target, "--user", user, "-a", "android.intent.action.VIEW", \
|
||||||
}
|
"-f", "0x18000020", "--es", "action", action
|
||||||
|
|
||||||
#define content_exec(...) content_exec_info(ctx.info.get(), __VA_ARGS__)
|
// 0x18000020 = FLAG_ACTIVITY_NEW_TASK|FLAG_ACTIVITY_MULTIPLE_TASK|FLAG_INCLUDE_STOPPED_PACKAGES
|
||||||
|
|
||||||
#define ex(s) "--extra", s
|
|
||||||
|
|
||||||
#define get_user(info) \
|
#define get_user(info) \
|
||||||
(info->cfg[SU_MULTIUSER_MODE] == MULTIUSER_MODE_USER \
|
(info->cfg[SU_MULTIUSER_MODE] == MULTIUSER_MODE_USER \
|
||||||
? info->uid / 100000 \
|
? info->uid / 100000 : 0)
|
||||||
: 0)
|
|
||||||
|
|
||||||
#define get_uid(info) \
|
#define get_uid(info) \
|
||||||
(info->cfg[SU_MULTIUSER_MODE] == MULTIUSER_MODE_OWNER_MANAGED \
|
(info->cfg[SU_MULTIUSER_MODE] == MULTIUSER_MODE_OWNER_MANAGED \
|
||||||
? info->uid % 100000 \
|
? info->uid % 100000 : info->uid)
|
||||||
: info->uid)
|
|
||||||
|
|
||||||
static const char *get_command(const su_request *to) {
|
#define get_cmd(to) \
|
||||||
if (to->command[0])
|
(to.command[0] ? to.command : to.shell[0] ? to.shell : DEFAULT_SHELL)
|
||||||
return to->command;
|
|
||||||
if (to->shell[0])
|
class Extra {
|
||||||
return to->shell;
|
const char *key;
|
||||||
return DEFAULT_SHELL;
|
enum {
|
||||||
|
INT,
|
||||||
|
BOOL,
|
||||||
|
STRING
|
||||||
|
} type;
|
||||||
|
union {
|
||||||
|
int int_val;
|
||||||
|
bool bool_val;
|
||||||
|
const char * str_val;
|
||||||
|
};
|
||||||
|
char i_buf[16];
|
||||||
|
char b_buf[32];
|
||||||
|
public:
|
||||||
|
Extra(const char *k, int v): key(k), type(INT), int_val(v) {}
|
||||||
|
Extra(const char *k, bool v): key(k), type(BOOL), bool_val(v) {}
|
||||||
|
Extra(const char *k, const char *v): key(k), type(STRING), str_val(v) {}
|
||||||
|
|
||||||
|
void add_intent(vector<const char *> &vec) {
|
||||||
|
const char *val;
|
||||||
|
switch (type) {
|
||||||
|
case INT:
|
||||||
|
vec.push_back("--ei");
|
||||||
|
sprintf(i_buf, "%d", int_val);
|
||||||
|
val = i_buf;
|
||||||
|
break;
|
||||||
|
case BOOL:
|
||||||
|
vec.push_back("--ez");
|
||||||
|
val = bool_val ? "true" : "false";
|
||||||
|
break;
|
||||||
|
case STRING:
|
||||||
|
vec.push_back("--es");
|
||||||
|
val = str_val;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
vec.push_back(key);
|
||||||
|
vec.push_back(val);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void exec_content_cmd(const char **args, const su_info *info) {
|
void add_bind(vector<const char *> &vec) {
|
||||||
|
switch (type) {
|
||||||
|
case INT:
|
||||||
|
sprintf(b_buf, "%s:i:%d", key, int_val);
|
||||||
|
break;
|
||||||
|
case BOOL:
|
||||||
|
sprintf(b_buf, "%s:b:%s", key, bool_val ? "true" : "false");
|
||||||
|
break;
|
||||||
|
case STRING:
|
||||||
|
sprintf(b_buf, "%s:s:%s", key, str_val);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
vec.push_back("--extra");
|
||||||
|
vec.push_back(b_buf);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
static bool check_error(int fd) {
|
||||||
|
char buf[1024];
|
||||||
|
unique_ptr<FILE, decltype(&fclose)> out(xfdopen(fd, "r"), fclose);
|
||||||
|
while (fgets(buf, sizeof(buf), out.get())) {
|
||||||
|
if (strncmp(buf, "Error", 5) == 0)
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void exec_cmd(const char *action, vector<Extra> &data,
|
||||||
|
const shared_ptr<su_info> &info, int mode = CONTENT_PROVIDER) {
|
||||||
char target[128];
|
char target[128];
|
||||||
sprintf(target, "content://%s.provider", info->str[SU_MANAGER].data());
|
|
||||||
char user[4];
|
char user[4];
|
||||||
sprintf(user, "%d", get_user(info));
|
sprintf(user, "%d", get_user(info));
|
||||||
|
|
||||||
// Fill in non static arguments
|
// First try content provider call method
|
||||||
args[5] = target;
|
if (mode >= CONTENT_PROVIDER) {
|
||||||
args[7] = user;
|
sprintf(target, "content://%s.provider", info->str[SU_MANAGER].data());
|
||||||
|
vector<const char *> args{ CALL_PROVIDER };
|
||||||
|
for (auto &e : data) {
|
||||||
|
e.add_bind(args);
|
||||||
|
}
|
||||||
|
args.push_back(nullptr);
|
||||||
exec_t exec {
|
exec_t exec {
|
||||||
.pre_exec = [] {
|
.err = true,
|
||||||
int null = xopen("/dev/null", O_WRONLY | O_CLOEXEC);
|
.fd = -1,
|
||||||
dup2(null, STDOUT_FILENO);
|
.pre_exec = [] { setenv("CLASSPATH", "/system/framework/content.jar", 1); },
|
||||||
dup2(null, STDERR_FILENO);
|
.argv = args.data()
|
||||||
setenv("CLASSPATH", "/system/framework/content.jar", 1);
|
|
||||||
},
|
|
||||||
.fork = fork_dont_care,
|
|
||||||
.argv = args
|
|
||||||
};
|
};
|
||||||
|
exec_command_sync(exec);
|
||||||
|
if (check_error(exec.fd))
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
vector<const char *> args{ START_ACTIVITY };
|
||||||
|
for (auto &e : data) {
|
||||||
|
e.add_intent(args);
|
||||||
|
}
|
||||||
|
args.push_back(nullptr);
|
||||||
|
exec_t exec {
|
||||||
|
.err = true,
|
||||||
|
.fd = -1,
|
||||||
|
.pre_exec = [] { setenv("CLASSPATH", "/system/framework/am.jar", 1); },
|
||||||
|
.argv = args.data()
|
||||||
|
};
|
||||||
|
|
||||||
|
if (mode >= PKG_ACTIVITY) {
|
||||||
|
// Then try start activity without component name
|
||||||
|
strcpy(target, info->str[SU_MANAGER].data());
|
||||||
|
exec_command_sync(exec);
|
||||||
|
if (check_error(exec.fd))
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Finally, fallback to start activity with component name
|
||||||
|
args[4] = "-n";
|
||||||
|
sprintf(target, "%s/a.m", info->str[SU_MANAGER].data());
|
||||||
|
exec.fd = -2;
|
||||||
|
exec.fork = fork_dont_care;
|
||||||
exec_command(exec);
|
exec_command(exec);
|
||||||
}
|
}
|
||||||
|
|
||||||
#define LOG_BODY \
|
|
||||||
"log", \
|
|
||||||
ex(fromUid), ex(toUid), ex(pid), ex(policy), \
|
|
||||||
ex(command.data()), ex(notify)
|
|
||||||
|
|
||||||
void app_log(const su_context &ctx) {
|
void app_log(const su_context &ctx) {
|
||||||
char fromUid[32];
|
if (fork_dont_care() == 0) {
|
||||||
sprintf(fromUid, "from.uid:i:%d", get_uid(ctx.info));
|
vector<Extra> extras;
|
||||||
|
extras.reserve(6);
|
||||||
|
extras.emplace_back("from.uid", get_uid(ctx.info));
|
||||||
|
extras.emplace_back("to.uid", ctx.req.uid);
|
||||||
|
extras.emplace_back("pid", ctx.pid);
|
||||||
|
extras.emplace_back("policy", ctx.info->access.policy);
|
||||||
|
extras.emplace_back("command", get_cmd(ctx.req));
|
||||||
|
extras.emplace_back("notify", (bool) ctx.info->access.notify);
|
||||||
|
|
||||||
char toUid[32];
|
exec_cmd("log", extras, ctx.info);
|
||||||
sprintf(toUid, "to.uid:i:%d", ctx.req.uid);
|
exit(0);
|
||||||
|
}
|
||||||
char pid[16];
|
|
||||||
sprintf(pid, "pid:i:%d", ctx.pid);
|
|
||||||
|
|
||||||
char policy[16];
|
|
||||||
sprintf(policy, "policy:i:%d", ctx.info->access.policy);
|
|
||||||
|
|
||||||
string command("command:s:");
|
|
||||||
command += get_command(&ctx.req);
|
|
||||||
|
|
||||||
char notify[16];
|
|
||||||
sprintf(notify, "notify:b:%s", ctx.info->access.notify ? "true" : "false");
|
|
||||||
|
|
||||||
content_exec(LOG_BODY)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#define NOTIFY_BODY \
|
|
||||||
"notify", ex(fromUid), ex(policy)
|
|
||||||
|
|
||||||
void app_notify(const su_context &ctx) {
|
void app_notify(const su_context &ctx) {
|
||||||
char fromUid[32];
|
if (fork_dont_care() == 0) {
|
||||||
sprintf(fromUid, "from.uid:i:%d", get_uid(ctx.info));
|
vector<Extra> extras;
|
||||||
|
extras.reserve(2);
|
||||||
|
extras.emplace_back("from.uid", get_uid(ctx.info));
|
||||||
|
extras.emplace_back("policy", ctx.info->access.policy);
|
||||||
|
|
||||||
char policy[16];
|
exec_cmd("notify", extras, ctx.info);
|
||||||
sprintf(policy, "policy:i:%d", ctx.info->access.policy);
|
exit(0);
|
||||||
|
}
|
||||||
content_exec(NOTIFY_BODY)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#define SOCKET_BODY \
|
|
||||||
"request", ex(sock)
|
|
||||||
|
|
||||||
void app_socket(const char *socket, const shared_ptr<su_info> &info) {
|
void app_socket(const char *socket, const shared_ptr<su_info> &info) {
|
||||||
char sock[128];
|
vector<Extra> extras;
|
||||||
sprintf(sock, "socket:s:%s", socket);
|
extras.reserve(1);
|
||||||
content_exec_info(info.get(), SOCKET_BODY)
|
extras.emplace_back("socket", socket);
|
||||||
|
|
||||||
|
exec_cmd("request", extras, info, PKG_ACTIVITY);
|
||||||
}
|
}
|
||||||
|
|
||||||
void socket_send_request(int fd, const shared_ptr<su_info> &info) {
|
void socket_send_request(int fd, const shared_ptr<su_info> &info) {
|
||||||
|
@ -17,7 +17,7 @@
|
|||||||
class su_info {
|
class su_info {
|
||||||
public:
|
public:
|
||||||
/* Unique key */
|
/* Unique key */
|
||||||
const unsigned uid;
|
const int uid;
|
||||||
|
|
||||||
/* These should be guarded with internal lock */
|
/* These should be guarded with internal lock */
|
||||||
db_settings cfg;
|
db_settings cfg;
|
||||||
@ -39,7 +39,7 @@ private:
|
|||||||
};
|
};
|
||||||
|
|
||||||
struct su_req_base {
|
struct su_req_base {
|
||||||
unsigned uid = UID_ROOT;
|
int uid = UID_ROOT;
|
||||||
bool login = false;
|
bool login = false;
|
||||||
bool keepenv = false;
|
bool keepenv = false;
|
||||||
bool mount_master = false;
|
bool mount_master = false;
|
||||||
@ -63,7 +63,7 @@ private:
|
|||||||
struct su_context {
|
struct su_context {
|
||||||
std::shared_ptr<su_info> info;
|
std::shared_ptr<su_info> info;
|
||||||
su_request req;
|
su_request req;
|
||||||
pid_t pid;
|
int pid;
|
||||||
};
|
};
|
||||||
|
|
||||||
void app_log(const su_context &ctx);
|
void app_log(const su_context &ctx);
|
||||||
|
@ -73,7 +73,8 @@ int strend(const char *s1, const char *s2) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
int exec_command(exec_t &exec) {
|
int exec_command(exec_t &exec) {
|
||||||
int pipefd[2] = {-1, -1}, outfd = -1;
|
int pipefd[] = {-1, -1};
|
||||||
|
int outfd = -1;
|
||||||
|
|
||||||
if (exec.fd == -1) {
|
if (exec.fd == -1) {
|
||||||
if (xpipe2(pipefd, O_CLOEXEC) == -1)
|
if (xpipe2(pipefd, O_CLOEXEC) == -1)
|
||||||
@ -113,10 +114,10 @@ int exec_command(exec_t &exec) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
int exec_command_sync(exec_t &exec) {
|
int exec_command_sync(exec_t &exec) {
|
||||||
int pid, status;
|
int pid = exec_command(exec);
|
||||||
pid = exec_command(exec);
|
|
||||||
if (pid < 0)
|
if (pid < 0)
|
||||||
return -1;
|
return -1;
|
||||||
|
int status;
|
||||||
waitpid(pid, &status, 0);
|
waitpid(pid, &status, 0);
|
||||||
return WEXITSTATUS(status);
|
return WEXITSTATUS(status);
|
||||||
}
|
}
|
||||||
|
@ -9,11 +9,11 @@
|
|||||||
android:icon="@drawable/ic_launcher"
|
android:icon="@drawable/ic_launcher"
|
||||||
android:installLocation="internalOnly"
|
android:installLocation="internalOnly"
|
||||||
android:label="Magisk Manager"
|
android:label="Magisk Manager"
|
||||||
android:supportsRtl="true">
|
android:supportsRtl="true"
|
||||||
|
android:theme="@android:style/Theme.Translucent.NoTitleBar">
|
||||||
|
|
||||||
<activity
|
<activity
|
||||||
android:name="a.r"
|
android:name="a.r"
|
||||||
android:theme="@android:style/Theme.Translucent.NoTitleBar"
|
|
||||||
android:process=":remote" />
|
android:process=":remote" />
|
||||||
|
|
||||||
<provider
|
<provider
|
||||||
|
@ -42,10 +42,15 @@
|
|||||||
|
|
||||||
<activity
|
<activity
|
||||||
android:name="a.m"
|
android:name="a.m"
|
||||||
android:theme="@android:style/Theme.Translucent.NoTitleBar"
|
|
||||||
android:directBootAware="true"
|
android:directBootAware="true"
|
||||||
android:excludeFromRecents="true"
|
android:excludeFromRecents="true"
|
||||||
android:exported="false" />
|
android:exported="false"
|
||||||
|
tools:ignore="AppLinkUrlError">
|
||||||
|
<intent-filter>
|
||||||
|
<action android:name="android.intent.action.VIEW"/>
|
||||||
|
<category android:name="android.intent.category.DEFAULT"/>
|
||||||
|
</intent-filter>
|
||||||
|
</activity>
|
||||||
|
|
||||||
<receiver
|
<receiver
|
||||||
android:name="a.h"
|
android:name="a.h"
|
||||||
|
Loading…
Reference in New Issue
Block a user