init_service.c 5.7 KB
Newer Older
1
/*
2
 * Copyright (c) 2020-2021 Huawei Device Co., Ltd.
3 4 5 6 7 8 9 10 11 12 13 14 15 16
 * 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 "init_service.h"

17
#include <dlfcn.h>
18 19 20 21
#include <stdlib.h>
#include <string.h>
#include <sys/param.h>
#include <sys/resource.h>
Y
yanmengzhao 已提交
22
#include <unistd.h>
23

24
#include "init_group_manager.h"
25 26 27
#include "init.h"
#include "init_log.h"
#include "init_param.h"
X
xionglei6 已提交
28
#include "init_utils.h"
29
#include "securec.h"
30 31
#include "token_setproc.h"
#include "nativetoken_kit.h"
X
xionglei6 已提交
32
#include "service_control.h"
33 34 35 36

#define MIN_IMPORTANT_LEVEL (-20)
#define MAX_IMPORTANT_LEVEL 19

X
xionglei6 已提交
37
void NotifyServiceChange(Service *service, int status)
38
{
X
xionglei6 已提交
39 40 41 42 43 44 45 46 47 48
    int size = 0;
    const InitArgInfo *statusMap = GetServieStatusMap(&size);
    INIT_ERROR_CHECK(statusMap != NULL && size > status, service->status = status;
        return, "Service status error %d", status);
    INIT_LOGI("NotifyServiceChange %s %s to %s", service->name,
        statusMap[service->status].name, statusMap[status].name);
    service->status = status;
    if (status == SERVICE_IDLE) {
        return;
    }
49
    char paramName[PARAM_NAME_LEN_MAX] = { 0 };
X
xionglei6 已提交
50 51 52 53 54
    int ret = snprintf_s(paramName, sizeof(paramName), sizeof(paramName) - 1,
        "%s.%s", STARTUP_SERVICE_CTL, service->name);
    if (ret >= 0) {
        SystemWriteParam(paramName, statusMap[status].name);
    }
55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77
}

int IsForbidden(const char *fieldStr)
{
    UNUSED(fieldStr);
    return 0;
}

int SetImportantValue(Service *service, const char *attrName, int value, int flag)
{
    UNUSED(attrName);
    UNUSED(flag);
    INIT_ERROR_CHECK(service != NULL, return SERVICE_FAILURE, "Set service attr failed! null ptr.");
    if (value >= MIN_IMPORTANT_LEVEL && value <= MAX_IMPORTANT_LEVEL) { // -20~19
        service->attribute |= SERVICE_ATTR_IMPORTANT;
        service->importance = value;
    } else {
        INIT_LOGE("Importance level = %d, is not between -20 and 19, error", value);
        return SERVICE_FAILURE;
    }
    return SERVICE_SUCCESS;
}

X
xionglei6 已提交
78
int ServiceExec(const Service *service)
79 80 81 82 83
{
    INIT_ERROR_CHECK(service != NULL && service->pathArgs.count > 0,
        return SERVICE_FAILURE, "Exec service failed! null ptr.");
    if (service->importance != 0) {
        if (setpriority(PRIO_PROCESS, 0, service->importance) != 0) {
X
xionglei6 已提交
84 85
            INIT_LOGE("setpriority failed for %s, importance = %d, err=%d",
                service->name, service->importance, errno);
86
                _exit(0x7f); // 0x7f: user specified
87 88
        }
    }
X
add ut  
xionglei6 已提交
89
    INIT_CHECK_ONLY_ELOG(unsetenv("UV_THREADPOOL_SIZE") == 0, "set UV_THREADPOOL_SIZE error : %d.", errno);
Y
yanmengzhao 已提交
90
#ifdef SUPPORT_PROFILER_HIDEBUG
Y
yanmengzhao 已提交
91 92 93 94 95
    do {
        if (access("/system/lib/libhidebug.so", F_OK) != 0) {
            INIT_LOGE("access failed, errno = %d\n", errno);
            break;
        }
Y
yanmengzhao 已提交
96 97 98
        void* handle = dlopen("/system/lib/libhidebug.so", RTLD_LAZY);
        if (handle == NULL) {
            INIT_LOGE("Failed to dlopen libhidebug.so, %s\n", dlerror());
Y
yanmengzhao 已提交
99
            break;
Y
yanmengzhao 已提交
100 101 102 103 104
        }
        bool (* initParam)();
        initParam = (bool (*)())dlsym(handle, "InitEnvironmentParam");
        if (initParam == NULL) {
            INIT_LOGE("Failed to dlsym InitEnvironmentParam, %s\n", dlerror());
Y
yanmengzhao 已提交
105 106
            dlclose(handle);
            break;
Y
yanmengzhao 已提交
107
        }
Y
yanmengzhao 已提交
108 109 110 111 112 113
        bool ret = (*initParam)(service->name);
        if (!ret) {
            INIT_LOGE("init parameters failed.\n");
        }
        dlclose(handle);
    } while (0);
Y
yanmengzhao 已提交
114
#endif
115
    // L2 Can not be reset env
X
xionglei6 已提交
116
    if (service->extraArgs.argv != NULL && service->extraArgs.count > 0) {
X
xionglei6 已提交
117
        INIT_CHECK_ONLY_ELOG(execv(service->extraArgs.argv[0], service->extraArgs.argv) == 0,
X
xionglei6 已提交
118 119 120 121 122
            "service %s execve failed! err %d.", service->name, errno);
    } else {
        INIT_CHECK_ONLY_ELOG(execv(service->pathArgs.argv[0], service->pathArgs.argv) == 0,
            "service %s execve failed! err %d.", service->name, errno);
    }
123
    return SERVICE_SUCCESS;
Q
Qin Fandong 已提交
124
}
125 126 127 128

int SetAccessToken(const Service *service)
{
    INIT_ERROR_CHECK(service != NULL, return SERVICE_FAILURE, "%s failed", service->name);
X
xlei1030 已提交
129 130 131
    int ret = SetSelfTokenID(service->tokenId);
    INIT_LOGI("%s: token id %lld, set token id result %d", service->name, service->tokenId, ret);
    return ret == 0 ? SERVICE_SUCCESS : SERVICE_FAILURE;
132 133 134 135 136 137 138 139 140 141 142 143
}

void GetAccessToken(void)
{
    InitGroupNode *node = GetNextGroupNode(NODE_TYPE_SERVICES, NULL);
    while (node != NULL) {
        Service *service = node->data.service;
        if (service != NULL) {
            if (service->capsArgs.count == 0) {
                service->capsArgs.argv = NULL;
            }
            if (strlen(service->apl) == 0) {
X
xionglei6 已提交
144
                (void)strncpy_s(service->apl, sizeof(service->apl), "system_core", sizeof(service->apl) - 1);
145 146 147 148
            }
            uint64_t tokenId = GetAccessTokenId(service->name, (const char **)service->capsArgs.argv,
                service->capsArgs.count, service->apl);
            if (tokenId  == 0) {
X
xionglei6 已提交
149
                INIT_LOGE("Get totken id %lld of service \' %s \' failed", tokenId, service->name);
150 151 152 153 154 155
            }
            service->tokenId = tokenId;
        }
        node = GetNextGroupNode(NODE_TYPE_SERVICES, node);
    }
}