param_manager.c 12.6 KB
Newer Older
Z
zhong_ning 已提交
1
/*
S
sun_fan 已提交
2
 * Copyright (c) 2021 Huawei Device Co., Ltd.
Z
zhong_ning 已提交
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 "param_manager.h"
17

Z
zhong_ning 已提交
18 19
#include <ctype.h>

S
sun_fan 已提交
20 21 22
#if !defined PARAM_SUPPORT_SELINUX && !defined PARAM_SUPPORT_DAC
static ParamSecurityLabel g_defaultSecurityLabel;
#endif
Z
zhong_ning 已提交
23

S
sun_fan 已提交
24
static int GetParamSecurityOps(ParamWorkSpace *workSpace, int isInit)
Z
zhong_ning 已提交
25
{
S
sun_fan 已提交
26 27
    UNUSED(isInit);
#if (defined PARAM_SUPPORT_SELINUX || defined PARAM_SUPPORT_DAC)
4
411148299@qq.com 已提交
28
    int ret = RegisterSecurityOps(&workSpace->paramSecurityOps, isInit);
S
sun_fan 已提交
29 30 31 32 33 34
    PARAM_CHECK(workSpace->paramSecurityOps.securityInitLabel != NULL, return -1, "Invalid securityInitLabel");
    ret = workSpace->paramSecurityOps.securityInitLabel(&workSpace->securityLabel, isInit);
    PARAM_CHECK(ret == 0, return PARAM_CODE_INVALID_NAME, "Failed to init security");
#else
    workSpace->securityLabel = &g_defaultSecurityLabel;
    workSpace->securityLabel->flags |= LABEL_ALL_PERMISSION;
Z
zhong_ning 已提交
35
#endif
4
411148299@qq.com 已提交
36
    return 0;
S
sun_fan 已提交
37
}
Z
zhong_ning 已提交
38

S
sun_fan 已提交
39
int InitParamWorkSpace(ParamWorkSpace *workSpace, int onlyRead)
Z
zhong_ning 已提交
40
{
S
sun_fan 已提交
41 42
    PARAM_CHECK(workSpace != NULL, return PARAM_CODE_INVALID_NAME, "Invalid param");
    if (PARAM_TEST_FLAG(workSpace->flags, WORKSPACE_FLAGS_INIT)) {
Z
zhong_ning 已提交
43 44
        return 0;
    }
S
sun_fan 已提交
45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65
    int isInit = 0;
    int op = DAC_READ;
    if (onlyRead == 0) {
        isInit = LABEL_INIT_FOR_INIT;
        op = DAC_WRITE;
    }
    int ret = GetParamSecurityOps(workSpace, isInit);
    PARAM_CHECK(ret == 0, return -1, "Failed to get security operations");
    ParamSecurityOps *paramSecurityOps = &workSpace->paramSecurityOps;
    if (!LABEL_IS_ALL_PERMITTED(workSpace->securityLabel)) {
        PARAM_CHECK(paramSecurityOps->securityFreeLabel != NULL, return -1, "Invalid securityFreeLabel");
        PARAM_CHECK(paramSecurityOps->securityCheckFilePermission != NULL, return -1, "Invalid securityCheck");
        PARAM_CHECK(paramSecurityOps->securityCheckParamPermission != NULL, return -1, "Invalid securityCheck");
        if (isInit == LABEL_INIT_FOR_INIT) {
            PARAM_CHECK(paramSecurityOps->securityGetLabel != NULL, return -1, "Invalid securityGetLabel");
            PARAM_CHECK(paramSecurityOps->securityDecodeLabel != NULL, return -1, "Invalid securityDecodeLabel");
        } else {
            PARAM_CHECK(paramSecurityOps->securityEncodeLabel != NULL, return -1, "Invalid securityEncodeLabel");
        }
        ret = paramSecurityOps->securityCheckFilePermission(workSpace->securityLabel, PARAM_STORAGE_PATH, op);
        PARAM_CHECK(ret == 0, return PARAM_CODE_INVALID_NAME, "No permission to read file %s", PARAM_STORAGE_PATH);
Z
zhong_ning 已提交
66
    }
67 68 69 70 71
    if (onlyRead) {
        ret = InitWorkSpace(CLIENT_PARAM_STORAGE_PATH, &workSpace->paramSpace, onlyRead);
    } else {
        ret = InitWorkSpace(PARAM_STORAGE_PATH, &workSpace->paramSpace, onlyRead);
    }
S
sun_fan 已提交
72 73
    PARAM_CHECK(ret == 0, return PARAM_CODE_INVALID_NAME, "Failed to init workspace");
    PARAM_SET_FLAG(workSpace->flags, WORKSPACE_FLAGS_INIT);
Z
zhong_ning 已提交
74 75 76 77 78
    return ret;
}

void CloseParamWorkSpace(ParamWorkSpace *workSpace)
{
X
xionglei6 已提交
79
     PARAM_CHECK(workSpace != NULL, return, "Invalid workSpace");
Z
zhong_ning 已提交
80
    CloseWorkSpace(&workSpace->paramSpace);
S
sun_fan 已提交
81 82 83 84
    if (workSpace->paramSecurityOps.securityFreeLabel != NULL) {
        workSpace->paramSecurityOps.securityFreeLabel(workSpace->securityLabel);
    }
    workSpace->flags = 0;
Z
zhong_ning 已提交
85 86
}

S
sun_fan 已提交
87
static uint32_t ReadCommitId(ParamNode *entry)
Z
zhong_ning 已提交
88
{
S
sun_fan 已提交
89 90 91 92
    uint32_t commitId = atomic_load_explicit(&entry->commitId, memory_order_acquire);
    while (commitId & PARAM_FLAGS_MODIFY) {
        futex_wait(&entry->commitId, commitId);
        commitId = atomic_load_explicit(&entry->commitId, memory_order_acquire);
Z
zhong_ning 已提交
93
    }
S
sun_fan 已提交
94
    return commitId & PARAM_FLAGS_COMMITID;
Z
zhong_ning 已提交
95 96
}

4
411148299@qq.com 已提交
97
int ReadParamCommitId(const ParamWorkSpace *workSpace, ParamHandle handle, uint32_t *commitId)
Z
zhong_ning 已提交
98
{
S
sun_fan 已提交
99 100
    PARAM_CHECK(workSpace != NULL && commitId != NULL, return PARAM_CODE_INVALID_PARAM, "Invalid workSpace");
    ParamNode *entry = (ParamNode *)GetTrieNode(&workSpace->paramSpace, handle);
Z
zhong_ning 已提交
101
    if (entry == NULL) {
S
sun_fan 已提交
102
        return -1;
Z
zhong_ning 已提交
103
    }
S
sun_fan 已提交
104
    *commitId = ReadCommitId(entry);
Z
zhong_ning 已提交
105 106 107
    return 0;
}

4
411148299@qq.com 已提交
108
int ReadParamWithCheck(const ParamWorkSpace *workSpace, const char *name, uint32_t op, ParamHandle *handle)
Z
zhong_ning 已提交
109
{
S
sun_fan 已提交
110 111
    PARAM_CHECK(handle != NULL, return PARAM_CODE_INVALID_PARAM, "Invalid param handle");
    PARAM_CHECK(workSpace != NULL && name != NULL, return PARAM_CODE_INVALID_PARAM, "Invalid param name");
4
411148299@qq.com 已提交
112
    *handle = -1;
S
sun_fan 已提交
113 114
    int ret = CheckParamPermission(workSpace, workSpace->securityLabel, name, op);
    PARAM_CHECK(ret == 0, return ret, "Forbid to access parameter %s", name);
Z
zhong_ning 已提交
115

S
sun_fan 已提交
116
    ParamTrieNode *node = FindTrieNode(&workSpace->paramSpace, name, strlen(name), NULL);
Z
zhong_ning 已提交
117 118 119 120
    if (node != NULL && node->dataIndex != 0) {
        *handle = node->dataIndex;
        return 0;
    }
S
sun_fan 已提交
121
    return PARAM_CODE_NOT_FOUND;
Z
zhong_ning 已提交
122 123
}

4
411148299@qq.com 已提交
124
int ReadParamValue(const ParamWorkSpace *workSpace, ParamHandle handle, char *value, uint32_t *length)
Z
zhong_ning 已提交
125
{
S
sun_fan 已提交
126 127
    PARAM_CHECK(workSpace != NULL && length != NULL, return PARAM_CODE_INVALID_PARAM, "Invalid param");
    ParamNode *entry = (ParamNode *)GetTrieNode(&workSpace->paramSpace, handle);
Z
zhong_ning 已提交
128 129 130 131
    if (entry == NULL) {
        return -1;
    }
    if (value == NULL) {
S
sun_fan 已提交
132
        *length = entry->valueLength + 1;
Z
zhong_ning 已提交
133 134
        return 0;
    }
S
sun_fan 已提交
135 136 137 138 139 140 141 142 143
    PARAM_CHECK(*length > entry->valueLength, return PARAM_CODE_INVALID_PARAM,
        "Invalid value len %u %u", *length, entry->valueLength);
    uint32_t commitId = ReadCommitId(entry);
    do {
        int ret = memcpy_s(value, *length, entry->data + entry->keyLength + 1, entry->valueLength);
        PARAM_CHECK(ret == EOK, return -1, "Failed to copy value");
        value[entry->valueLength] = '\0';
        *length = entry->valueLength;
    } while (commitId != ReadCommitId(entry));
Z
zhong_ning 已提交
144 145 146
    return 0;
}

4
411148299@qq.com 已提交
147
int ReadParamName(const ParamWorkSpace *workSpace, ParamHandle handle, char *name, uint32_t length)
Z
zhong_ning 已提交
148 149
{
    PARAM_CHECK(workSpace != NULL && name != NULL, return PARAM_CODE_INVALID_PARAM, "Invalid param");
S
sun_fan 已提交
150
    ParamNode *entry = (ParamNode *)GetTrieNode(&workSpace->paramSpace, handle);
Z
zhong_ning 已提交
151 152 153
    if (entry == NULL) {
        return -1;
    }
S
sun_fan 已提交
154 155 156 157 158
    PARAM_CHECK(length > entry->keyLength, return -1, "Invalid param size %u %u", entry->keyLength, length);
    int ret = memcpy_s(name, length, entry->data, entry->keyLength);
    PARAM_CHECK(ret == EOK, return PARAM_CODE_INVALID_PARAM, "Failed to copy name");
    name[entry->keyLength] = '\0';
    return 0;
Z
zhong_ning 已提交
159 160 161 162
}

int CheckParamName(const char *name, int info)
{
4
411148299@qq.com 已提交
163
    PARAM_CHECK(name != NULL, return PARAM_CODE_INVALID_PARAM, "Invalid param");
Z
zhong_ning 已提交
164
    size_t nameLen = strlen(name);
S
sun_fan 已提交
165
    if (nameLen >= PARAM_NAME_LEN_MAX) {
Z
zhong_ning 已提交
166 167
        return PARAM_CODE_INVALID_NAME;
    }
S
sun_fan 已提交
168 169 170
    if (strcmp(name, "#") == 0) {
        return 0;
    }
Z
zhong_ning 已提交
171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196

    if (nameLen < 1 || name[0] == '.' || (!info && name[nameLen - 1] == '.')) {
        PARAM_LOGE("CheckParamName %s %d", name, info);
        return PARAM_CODE_INVALID_NAME;
    }

    /* Only allow alphanumeric, plus '.', '-', '@', ':', or '_' */
    /* Don't allow ".." to appear in a param name */
    for (size_t i = 0; i < nameLen; i++) {
        if (name[i] == '.') {
            if (name[i - 1] == '.') {
                return PARAM_CODE_INVALID_NAME;
            }
            continue;
        }
        if (name[i] == '_' || name[i] == '-' || name[i] == '@' || name[i] == ':') {
            continue;
        }
        if (isalnum(name[i])) {
            continue;
        }
        return PARAM_CODE_INVALID_NAME;
    }
    return 0;
}

4
411148299@qq.com 已提交
197
static int ProcessParamTraversal(const WorkSpace *workSpace, const ParamTrieNode *node, void *cookie)
Z
zhong_ning 已提交
198
{
S
sun_fan 已提交
199 200 201 202 203
    UNUSED(workSpace);
    ParamTraversalContext *context = (ParamTraversalContext *)cookie;
    ParamTrieNode *current = (ParamTrieNode *)node;
    if (current == NULL) {
        return 0;
Z
zhong_ning 已提交
204
    }
S
sun_fan 已提交
205 206 207 208
    if (current->dataIndex == 0) {
        return 0;
    }
    context->traversalParamPtr(current->dataIndex, context->context);
Z
zhong_ning 已提交
209 210 211
    return 0;
}

4
411148299@qq.com 已提交
212
int TraversalParam(const ParamWorkSpace *workSpace, TraversalParamPtr walkFunc, void *cookie)
Z
zhong_ning 已提交
213
{
4
411148299@qq.com 已提交
214
    PARAM_CHECK(workSpace != NULL && walkFunc != NULL, return PARAM_CODE_INVALID_PARAM, "Invalid param");
S
sun_fan 已提交
215 216 217 218
    ParamTraversalContext context = {
        walkFunc, cookie
    };
    return TraversalTrieNode(&workSpace->paramSpace, NULL, ProcessParamTraversal, &context);
Z
zhong_ning 已提交
219 220
}

4
411148299@qq.com 已提交
221 222
int CheckParamPermission(const ParamWorkSpace *workSpace,
    const ParamSecurityLabel *srcLabel, const char *name, uint32_t mode)
Z
zhong_ning 已提交
223
{
4
411148299@qq.com 已提交
224 225
    PARAM_CHECK(workSpace != NULL && workSpace->securityLabel != NULL,
        return PARAM_CODE_INVALID_PARAM, "Invalid param");
S
sun_fan 已提交
226 227
    if (LABEL_IS_ALL_PERMITTED(workSpace->securityLabel)) {
        return 0;
Z
zhong_ning 已提交
228
    }
S
sun_fan 已提交
229 230 231 232 233 234 235 236
    PARAM_CHECK(name != NULL && srcLabel != NULL, return -1, "Invalid param");
    if (workSpace->paramSecurityOps.securityCheckParamPermission == NULL) {
        return DAC_RESULT_FORBIDED;
    }
    uint32_t labelIndex = 0;
    FindTrieNode(&workSpace->paramSpace, name, strlen(name), &labelIndex);
    ParamSecruityNode *node = (ParamSecruityNode *)GetTrieNode(&workSpace->paramSpace, labelIndex);
    PARAM_CHECK(node != NULL, return DAC_RESULT_FORBIDED, "Can not get security label %d", labelIndex);
Z
zhong_ning 已提交
237

S
sun_fan 已提交
238
    ParamAuditData auditData = {};
Z
zhong_ning 已提交
239
    auditData.name = name;
S
sun_fan 已提交
240 241 242 243 244
    auditData.dacData.uid = node->uid;
    auditData.dacData.gid = node->gid;
    auditData.dacData.mode = node->mode;
    auditData.label = node->data;
    return workSpace->paramSecurityOps.securityCheckParamPermission(srcLabel, &auditData, mode);
Z
zhong_ning 已提交
245 246
}

4
411148299@qq.com 已提交
247
static int DumpTrieDataNodeTraversal(const WorkSpace *workSpace, const ParamTrieNode *node, void *cookie)
Z
zhong_ning 已提交
248
{
S
sun_fan 已提交
249 250 251 252
    int verbose = *(int *)cookie;
    ParamTrieNode *current = (ParamTrieNode *)node;
    if (current == NULL) {
        return 0;
Z
zhong_ning 已提交
253
    }
S
sun_fan 已提交
254
    if (verbose) {
4
411148299@qq.com 已提交
255
        printf("\tTrie node info [%u,%u,%u] data: %u label: %u key length:%d \n\t  key: %s \n",
S
sun_fan 已提交
256 257
            current->left, current->right, current->child,
            current->dataIndex, current->labelIndex, current->length, current->key);
Z
zhong_ning 已提交
258
    }
S
sun_fan 已提交
259 260 261
    if (current->dataIndex != 0) {
        ParamNode *entry = (ParamNode *)GetTrieNode(workSpace, current->dataIndex);
        if (entry != NULL) {
4
411148299@qq.com 已提交
262
            printf("\tparameter length info [%d, %d] \n\t  param: %s \n",
S
sun_fan 已提交
263
                entry->keyLength, entry->valueLength, (entry != NULL) ? entry->data : "null");
Z
zhong_ning 已提交
264 265
        }
    }
4
411148299@qq.com 已提交
266
    if (current->labelIndex != 0 && verbose) {
S
sun_fan 已提交
267 268
        ParamSecruityNode *label = (ParamSecruityNode *)GetTrieNode(workSpace, current->labelIndex);
        if (label != NULL) {
4
411148299@qq.com 已提交
269
            printf("\tparameter label dac %d %d %o \n\t  label: %s \n",
S
sun_fan 已提交
270 271
                label->uid, label->gid, label->mode, (label->length > 0) ? label->data : "null");
        }
Z
zhong_ning 已提交
272
    }
S
sun_fan 已提交
273
    return 0;
Z
zhong_ning 已提交
274 275
}

4
411148299@qq.com 已提交
276
static void DumpWorkSpace(const ParamWorkSpace *workSpace, int verbose)
Z
zhong_ning 已提交
277
{
S
sun_fan 已提交
278 279
    printf("workSpace information \n");
    printf("    map file: %s \n", workSpace->paramSpace.fileName);
280 281 282 283 284 285 286 287
    if (workSpace->paramSpace.area != NULL) {
        printf("    total size: %d \n", workSpace->paramSpace.area->dataSize);
        printf("    first offset: %d \n", workSpace->paramSpace.area->firstNode);
        printf("    current offset: %d \n", workSpace->paramSpace.area->currOffset);
        printf("    total node: %d \n", workSpace->paramSpace.area->trieNodeCount);
        printf("    total param node: %d \n", workSpace->paramSpace.area->paramNodeCount);
        printf("    total security node: %d\n", workSpace->paramSpace.area->securityNodeCount);
    }
S
sun_fan 已提交
288 289
    printf("    node info: \n");
    TraversalTrieNode(&workSpace->paramSpace, NULL, DumpTrieDataNodeTraversal, (void *)&verbose);
Z
zhong_ning 已提交
290 291
}

4
411148299@qq.com 已提交
292
void DumpParameters(const ParamWorkSpace *workSpace, int verbose)
Z
zhong_ning 已提交
293
{
4
411148299@qq.com 已提交
294
    PARAM_CHECK(workSpace != NULL && workSpace->securityLabel != NULL, return, "Invalid param");
S
sun_fan 已提交
295 296 297 298 299 300 301 302 303 304
    printf("Dump all paramters begin ...\n");
    DumpWorkSpace(workSpace, verbose);
    if (verbose) {
        printf("Local sercurity information\n");
        printf("\t pid: %d uid: %d gid: %d \n",
            workSpace->securityLabel->cred.pid,
            workSpace->securityLabel->cred.uid,
            workSpace->securityLabel->cred.gid);
    }
    printf("Dump all paramters finish\n");
Z
zhong_ning 已提交
305
}