/* * Copyright (c) 2024 Huawei Device Co., Ltd. * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #include "appspawn_server.h" #undef _GNU_SOURCE #define _GNU_SOURCE #include #include #include #include "appspawn_trace.h" #include "appspawn_utils.h" #ifndef OHOS_LITE #include "appspawn_manager.h" #endif #define MAX_FORK_TIME (30 * 1000) // 30ms static void NotifyResToParent(struct AppSpawnContent *content, AppSpawnClient *client, int result) { APPSPAWN_LOGI("NotifyResToParent: %{public}d", result); if (content->notifyResToParent != NULL) { content->notifyResToParent(content, client, result); } } void ProcessExit(int code) { APPSPAWN_LOGI("App exit code: %{public}d", code); #ifdef OHOS_LITE _exit(0x7f); // 0x7f user exit #else quick_exit(0); #endif } #ifdef APPSPAWN_HELPER __attribute__((visibility("default"))) _Noreturn void exit(int code) { char *checkExit = getenv(APPSPAWN_CHECK_EXIT); if (checkExit && atoi(checkExit) == getpid()) { APPSPAWN_LOGF("Unexpected call: exit(%{public}d)", code); abort(); } // hook `exit` to `ProcessExit` to ensure app exit in a clean way ProcessExit(code); // should not come here abort(); } #endif #ifdef USE_ENCAPS #include #define OH_ENCAPS_PROC_TYPE_BASE 0x18 #define OH_ENCAPS_MAGIC 'E' #define OH_PROC_APP 4 #define SET_PROC_TYPE_CMD _IOW(OH_ENCAPS_MAGIC, OH_ENCAPS_PROC_TYPE_BASE, uint32_t) static void SetEncapsFlag() { uint32_t flag = OH_PROC_APP; int fdEncaps = open("/dev/encaps", O_RDWR); if (fdEncaps < 0) { APPSPAWN_LOGE("AppSpawnChild open encaps failed"); return; } int ret = ioctl(fdEncaps, SET_PROC_TYPE_CMD, &flag); if (ret != 0) { APPSPAWN_LOGE("AppSpawnChild SetEncapsFlag failed"); } close(fdEncaps); } #endif int AppSpawnChild(AppSpawnContent *content, AppSpawnClient *client) { APPSPAWN_CHECK(content != NULL && client != NULL, return -1, "Invalid arg for appspawn child"); APPSPAWN_LOGI("AppSpawnChild id %{public}u flags: 0x%{public}x", client->id, client->flags); StartAppspawnTrace("AppSpawnExecuteClearEnvHook"); int ret = AppSpawnExecuteClearEnvHook(content, client); FinishAppspawnTrace(); APPSPAWN_CHECK_ONLY_EXPER(ret == 0, NotifyResToParent(content, client, ret); AppSpawnEnvClear(content, client); return 0); if (client->flags & APP_COLD_START) { // cold start fail, to start normal if (content->coldStartApp != NULL && content->coldStartApp(content, client) == 0) { return 0; } APPSPAWN_LOGW("AppSpawnChild cold start fail %{public}u", client->id); } #ifdef USE_ENCAPS SetEncapsFlag(); #endif StartAppspawnTrace("AppSpawnExecuteSpawningHook"); ret = AppSpawnExecuteSpawningHook(content, client); FinishAppspawnTrace(); APPSPAWN_CHECK_ONLY_EXPER(ret == 0, NotifyResToParent(content, client, ret); AppSpawnEnvClear(content, client); return 0); StartAppspawnTrace("AppSpawnExecutePreReplyHook"); ret = AppSpawnExecutePreReplyHook(content, client); FinishAppspawnTrace(); APPSPAWN_CHECK_ONLY_EXPER(ret == 0, NotifyResToParent(content, client, ret); AppSpawnEnvClear(content, client); return 0); // notify success to father process and start app process StartAppspawnTrace("NotifyResToParent"); NotifyResToParent(content, client, 0); FinishAppspawnTrace(); StartAppspawnTrace("AppSpawnExecutePostReplyHook"); (void)AppSpawnExecutePostReplyHook(content, client); FinishAppspawnTrace(); if (content->runChildProcessor != NULL) { ret = content->runChildProcessor(content, client); } if (ret != 0) { AppSpawnEnvClear(content, client); } return 0; } static int CloneAppSpawn(void *arg) { APPSPAWN_CHECK(arg != NULL, return -1, "Invalid content for appspawn"); AppSpawnForkArg *forkArg = (AppSpawnForkArg *)arg; ProcessExit(AppSpawnChild(forkArg->content, forkArg->client)); return 0; } #ifndef OHOS_LITE static void NwebSpawnCloneChildProcess(AppSpawnContent *content, AppSpawnClient *client, pid_t *pid) { AppSpawnForkArg arg; arg.client = client; arg.content = content; #ifndef APPSPAWN_TEST AppSpawningCtx *property = (AppSpawningCtx *)client; uint32_t len = 0; char *processType = (char *)(GetAppSpawnMsgExtInfo(property->message, MSG_EXT_NAME_PROCESS_TYPE, &len)); APPSPAWN_CHECK(processType != NULL, return, "Invalid processType data"); if (strcmp(processType, "gpu") == 0) { *pid = clone(CloneAppSpawn, NULL, CLONE_NEWNET | SIGCHLD, (void *)&arg); } else { *pid = clone(CloneAppSpawn, NULL, content->sandboxNsFlags | SIGCHLD, (void *)&arg); } #else *pid = clone(CloneAppSpawn, NULL, content->sandboxNsFlags | SIGCHLD, (void *)&arg); #endif } #endif static void AppSpawnForkChildProcess(AppSpawnContent *content, AppSpawnClient *client, pid_t *pid) { struct timespec forkStart = {0}; clock_gettime(CLOCK_MONOTONIC, &forkStart); StartAppspawnTrace("AppspawnForkStart"); *pid = fork(); if (*pid == 0) { struct timespec forkEnd = {0}; clock_gettime(CLOCK_MONOTONIC, &forkEnd); uint64_t diff = DiffTime(&forkStart, &forkEnd); APPSPAWN_CHECK_ONLY_LOG(diff < MAX_FORK_TIME, "fork time %{public}" PRId64 " us", diff); ProcessExit(AppSpawnChild(content, client)); } else { FinishAppspawnTrace(); } } int AppSpawnProcessMsg(AppSpawnContent *content, AppSpawnClient *client, pid_t *childPid) { APPSPAWN_CHECK(content != NULL, return -1, "Invalid content for appspawn"); APPSPAWN_CHECK(client != NULL && childPid != NULL, return -1, "Invalid client for appspawn"); APPSPAWN_LOGI("AppSpawnProcessMsg id: %{public}d mode: %{public}d sandboxNsFlags: 0x%{public}x", client->id, content->mode, content->sandboxNsFlags); pid_t pid = 0; #ifndef OHOS_LITE if (content->mode == MODE_FOR_NWEB_SPAWN) { NwebSpawnCloneChildProcess(content, client, &pid); } else { #else { #endif AppSpawnForkChildProcess(content, client, &pid); } APPSPAWN_CHECK(pid >= 0, return APPSPAWN_FORK_FAIL, "fork child process error: %{public}d", errno); *childPid = pid; return 0; }