param_selinux.c 9.3 KB
Newer Older
S
sun_fan 已提交
1 2 3 4 5 6 7 8 9 10 11 12 13 14
/*
 * Copyright (c) 2021 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.
 */
M
Mupceet 已提交
15 16
#include <dlfcn.h>
#include <sys/socket.h>
S
sun_fan 已提交
17

18
#include "init_utils.h"
M
Mupceet 已提交
19
#include "param_manager.h"
S
sun_fan 已提交
20 21
#include "param_security.h"
#include "param_utils.h"
M
Mupceet 已提交
22
#include "param_base.h"
M
Mupceet 已提交
23 24
#ifdef PARAM_SUPPORT_SELINUX
#include "selinux_parameter.h"
M
Mupceet 已提交
25 26
#include <policycoreutils.h>
#include <selinux/selinux.h>
M
Mupceet 已提交
27
#endif
S
sun_fan 已提交
28

M
Mupceet 已提交
29 30
#ifdef __aarch64__
#define CHECKER_LIB_NAME "/system/lib64/libparaperm_checker.z.so"
U
unknown 已提交
31
#define CHECKER_UPDATER_LIB_NAME "/lib64/libparaperm_checker.z.so"
M
Mupceet 已提交
32 33
#else
#define CHECKER_LIB_NAME "/system/lib/libparaperm_checker.z.so"
U
unknown 已提交
34
#define CHECKER_UPDATER_LIB_NAME "/lib/libparaperm_checker.z.so"
M
Mupceet 已提交
35 36
#endif

M
Mupceet 已提交
37 38
static SelinuxSpace g_selinuxSpace = {0};
static int InitLocalSecurityLabel(ParamSecurityLabel *security, int isInit)
S
sun_fan 已提交
39 40
{
    UNUSED(isInit);
M
Mupceet 已提交
41 42 43 44 45
    PARAM_CHECK(security != NULL, return -1, "Invalid security");
    security->cred.pid = getpid();
    security->cred.uid = geteuid();
    security->cred.gid = getegid();
    security->flags[PARAM_SECURITY_SELINUX] = 0;
M
Mupceet 已提交
46
    PARAM_LOGV("InitLocalSecurityLabel");
M
Mupceet 已提交
47 48
#if !(defined STARTUP_INIT_TEST || defined LOCAL_TEST)
    if (g_selinuxSpace.selinuxHandle == NULL) {
U
unknown 已提交
49 50
        const char *libname = (InUpdaterMode() == 1) ? CHECKER_UPDATER_LIB_NAME : CHECKER_LIB_NAME;
        g_selinuxSpace.selinuxHandle = dlopen(libname, RTLD_LAZY);
M
Mupceet 已提交
51
        PARAM_CHECK(g_selinuxSpace.selinuxHandle != NULL,
M
Mupceet 已提交
52
            return 0, "Failed to dlsym selinuxHandle, %s", dlerror());
M
Mupceet 已提交
53 54 55 56 57 58 59
    }
    void *handle = g_selinuxSpace.selinuxHandle;
    if (g_selinuxSpace.setParamCheck == NULL) {
        g_selinuxSpace.setParamCheck = (SelinuxSetParamCheck)dlsym(handle, "SetParamCheck");
        PARAM_CHECK(g_selinuxSpace.setParamCheck != NULL, return -1, "Failed to dlsym setParamCheck %s", dlerror());
    }
    if (g_selinuxSpace.getParamList == NULL) {
M
Mupceet 已提交
60
        g_selinuxSpace.getParamList = (ParamContextsList *(*)()) dlsym(handle, "GetParamList");
M
Mupceet 已提交
61 62 63
        PARAM_CHECK(g_selinuxSpace.getParamList != NULL, return -1, "Failed to dlsym getParamList %s", dlerror());
    }
    if (g_selinuxSpace.getParamLabel == NULL) {
M
Mupceet 已提交
64
        g_selinuxSpace.getParamLabel = (const char * (*)(const char *))dlsym(handle, "GetParamLabel");
M
Mupceet 已提交
65 66
        PARAM_CHECK(g_selinuxSpace.getParamLabel != NULL, return -1, "Failed to dlsym getParamLabel %s", dlerror());
    }
M
Mupceet 已提交
67 68 69
    if (g_selinuxSpace.initParamSelinux == NULL) {
        g_selinuxSpace.initParamSelinux = (void (*)())dlsym(handle, "InitParamSelinux");
        PARAM_CHECK(g_selinuxSpace.initParamSelinux != NULL, return -1, "Failed to dlsym initParamSelinux ");
M
Mupceet 已提交
70
    }
M
Mupceet 已提交
71 72 73
    if (g_selinuxSpace.readParamCheck == NULL) {
        g_selinuxSpace.readParamCheck = (int (*)(const char *))dlsym(handle, "ReadParamCheck");
    }
M
Mupceet 已提交
74 75 76 77 78 79
    if (g_selinuxSpace.destroyParamList == NULL) {
        g_selinuxSpace.destroyParamList =
            (void (*)(ParamContextsList **))dlsym(handle, "DestroyParamList");
        PARAM_CHECK(g_selinuxSpace.destroyParamList != NULL,
            return -1, "Failed to dlsym destroyParamList %s", dlerror());
    }
M
Mupceet 已提交
80
    g_selinuxSpace.initParamSelinux();
M
Mupceet 已提交
81
#endif
M
Mupceet 已提交
82
    PARAM_LOGV("Load selinux lib success.");
S
sun_fan 已提交
83 84 85 86 87 88 89 90
    return 0;
}

static int FreeLocalSecurityLabel(ParamSecurityLabel *srcLabel)
{
    return 0;
}

M
Mupceet 已提交
91 92 93 94 95 96 97 98 99 100 101 102 103
static void SetSelinuxFileCon(const char *name, const char *context)
{
    static char buffer[FILENAME_LEN_MAX] = {0};
    int len = ParamSprintf(buffer, sizeof(buffer), "%s/%s", PARAM_STORAGE_PATH, context);
    if (len > 0) {
        buffer[len] = '\0';
        PARAM_LOGI("setfilecon name %s path: %s %s ", name, context, buffer);
        if (setfilecon(buffer, context) < 0) {
            PARAM_LOGE("Failed to setfilecon %s ", context);
        }
    }
}

104
static int SelinuxGetAllLabel(int readOnly)
S
sun_fan 已提交
105
{
M
Mupceet 已提交
106 107 108
    PARAM_CHECK(g_selinuxSpace.getParamList != NULL, return DAC_RESULT_FORBIDED, "Invalid getParamList");
    ParamContextsList *head = g_selinuxSpace.getParamList();
    ParamContextsList *node = head;
M
Mupceet 已提交
109

M
Mupceet 已提交
110 111 112 113 114
    int count = 0;
    while (node != NULL) {
        PARAM_LOGV("GetParamSecurityLabel name %s content %s", node->info.paraName, node->info.paraContext);
        if (node->info.paraContext == NULL || node->info.paraName == NULL) {
            node = node->next;
S
sun_fan 已提交
115 116
            continue;
        }
117
        int ret = AddWorkSpace(node->info.paraContext, readOnly, PARAM_WORKSPACE_DEF);
M
Mupceet 已提交
118 119 120 121 122
        if (ret != 0) {
            PARAM_LOGE("Forbid to add selinux workspace %s %s", node->info.paraName, node->info.paraContext);
            node = node->next;
            continue;
        }
M
Mupceet 已提交
123
        count++;
M
Mupceet 已提交
124 125 126 127 128 129 130
        if (readOnly != 0) {
            node = node->next;
            continue;
        }
        // set selinx label
        SetSelinuxFileCon(node->info.paraName, node->info.paraContext);
        node = node->next;
4
411148299@qq.com 已提交
131
    }
M
Mupceet 已提交
132

133
    int ret = AddWorkSpace(WORKSPACE_NAME_DEF_SELINUX, readOnly, PARAM_WORKSPACE_MAX);
M
Mupceet 已提交
134
    PARAM_CHECK(ret == 0, return -1,
M
Mupceet 已提交
135
        "Failed to add selinux workspace %s", WORKSPACE_NAME_DEF_SELINUX);
M
Mupceet 已提交
136 137 138
    if (readOnly == 0) {
        SetSelinuxFileCon(WORKSPACE_NAME_DEF_SELINUX, WORKSPACE_NAME_DEF_SELINUX);
    }
M
Mupceet 已提交
139
    PARAM_LOGI("SelinuxGetAllLabel count %d", count);
M
Mupceet 已提交
140
    return 0;
S
sun_fan 已提交
141 142
}

143 144 145 146 147 148
static int SelinuxGetParamSecurityLabel(const char *path)
{
    UNUSED(path);
    return SelinuxGetAllLabel(0);
}

S
sun_fan 已提交
149 150 151 152 153 154 155
static int CheckFilePermission(const ParamSecurityLabel *localLabel, const char *fileName, int flags)
{
    UNUSED(flags);
    PARAM_CHECK(localLabel != NULL && fileName != NULL, return -1, "Invalid param");
    return 0;
}

156 157
static int SelinuxReadParamCheck(const char *name)
{
158
    int ret = DAC_RESULT_FORBIDED;
M
Mupceet 已提交
159 160 161 162
    if (g_selinuxSpace.readParamCheck != NULL) {
        ret = g_selinuxSpace.readParamCheck(name);
        PARAM_LOGI("SelinuxReadParamCheck name %s ret %d", name, ret);
    }
M
Mupceet 已提交
163
    const char *label = GetSelinuxContent(name);
164 165 166 167 168
    if (label == NULL) { // open file with readonly
        ret = AddWorkSpace(WORKSPACE_NAME_DEF_SELINUX, 1, PARAM_WORKSPACE_MAX);
    } else {
        ret = AddWorkSpace(label, 1, PARAM_WORKSPACE_MAX);
    }
169
    if (ret != 0) {
M
Mupceet 已提交
170
        PARAM_LOGV("SelinuxReadParamCheck name %s label %s ", name, label);
171 172 173 174 175
        return DAC_RESULT_FORBIDED;
    }
    return DAC_RESULT_PERMISSION;
}

M
Mupceet 已提交
176
static int SelinuxCheckParamPermission(const ParamSecurityLabel *srcLabel, const char *name, uint32_t mode)
S
sun_fan 已提交
177
{
M
Mupceet 已提交
178 179 180 181 182 183 184 185 186 187
    int ret = DAC_RESULT_FORBIDED;
    PARAM_CHECK(g_selinuxSpace.setParamCheck != NULL, return ret, "Invalid setParamCheck");
    // check
    struct ucred uc;
    uc.pid = srcLabel->cred.pid;
    uc.uid = srcLabel->cred.uid;
    uc.gid = srcLabel->cred.gid;
    if (mode == DAC_WRITE) {
        ret = g_selinuxSpace.setParamCheck(name, &uc);
    } else {
L
laiguizhong 已提交
188
#ifndef STARTUP_INIT_TEST
189
        ret = SelinuxReadParamCheck(name);
L
laiguizhong 已提交
190 191 192
#else
        ret = g_selinuxSpace.readParamCheck(name);
#endif
M
Mupceet 已提交
193 194
    }
    if (ret != 0) {
M
Mupceet 已提交
195 196 197 198
        PARAM_LOGW("Selinux check name %s pid %d uid %d %d result %d", name, uc.pid, uc.uid, uc.gid, ret);
        ret = DAC_RESULT_FORBIDED;
    } else {
        ret = DAC_RESULT_PERMISSION;
M
Mupceet 已提交
199 200
    }
    return ret;
S
sun_fan 已提交
201 202
}

M
Mupceet 已提交
203 204 205 206 207
static int UpdaterCheckParamPermission(const ParamSecurityLabel *srcLabel, const char *name, uint32_t mode)
{
    return DAC_RESULT_PERMISSION;
}

M
Mupceet 已提交
208
INIT_LOCAL_API int RegisterSecuritySelinuxOps(ParamSecurityOps *ops, int isInit)
S
sun_fan 已提交
209 210
{
    PARAM_CHECK(ops != NULL, return -1, "Invalid param");
M
Mupceet 已提交
211
    int ret = ParamStrCpy(ops->name, sizeof(ops->name), "selinux");
S
sun_fan 已提交
212 213 214
    ops->securityGetLabel = NULL;
    ops->securityInitLabel = InitLocalSecurityLabel;
    ops->securityCheckFilePermission = CheckFilePermission;
M
Mupceet 已提交
215 216 217 218 219
    if (InUpdaterMode() == 1) {
        ops->securityCheckParamPermission = UpdaterCheckParamPermission;
    } else {
        ops->securityCheckParamPermission = SelinuxCheckParamPermission;
    }
S
sun_fan 已提交
220
    ops->securityFreeLabel = FreeLocalSecurityLabel;
221
    if (isInit != 0) {
M
Mupceet 已提交
222
        ops->securityGetLabel = SelinuxGetParamSecurityLabel;
S
sun_fan 已提交
223
    }
M
Mupceet 已提交
224
    return ret;
S
sun_fan 已提交
225
}
M
Mupceet 已提交
226

M
Mupceet 已提交
227
INIT_INNER_API const char *GetSelinuxContent(const char *name)
M
Mupceet 已提交
228
{
M
Mupceet 已提交
229 230 231 232 233 234
    if (g_selinuxSpace.getParamLabel != NULL) {
        return g_selinuxSpace.getParamLabel(name);
    } else {
        PARAM_LOGE("Can not init selinux");
        return WORKSPACE_NAME_DEF_SELINUX;
    }
M
Mupceet 已提交
235 236
}

M
Mupceet 已提交
237
INIT_LOCAL_API void OpenPermissionWorkSpace(void)
238 239 240 241 242
{
    // open workspace by readonly
    SelinuxGetAllLabel(1);
}

M
Mupceet 已提交
243 244
#if defined STARTUP_INIT_TEST || defined LOCAL_TEST
void SetSelinuxOps(const SelinuxSpace *space)
S
sun_fan 已提交
245
{
M
Mupceet 已提交
246 247 248 249 250 251
    g_selinuxSpace.setSelinuxLogCallback = space->setSelinuxLogCallback;
    g_selinuxSpace.setParamCheck = space->setParamCheck;
    g_selinuxSpace.getParamLabel = space->getParamLabel;
    g_selinuxSpace.readParamCheck = space->readParamCheck;
    g_selinuxSpace.getParamList = space->getParamList;
    g_selinuxSpace.destroyParamList = space->destroyParamList;
S
sun_fan 已提交
252 253
}
#endif