1bae44755Sopenharmony_ci/*
2bae44755Sopenharmony_ci * Copyright (C) 2022 Huawei Device Co., Ltd.
3bae44755Sopenharmony_ci * Licensed under the Apache License, Version 2.0 (the "License");
4bae44755Sopenharmony_ci * you may not use this file except in compliance with the License.
5bae44755Sopenharmony_ci * You may obtain a copy of the License at
6bae44755Sopenharmony_ci *
7bae44755Sopenharmony_ci *     http://www.apache.org/licenses/LICENSE-2.0
8bae44755Sopenharmony_ci *
9bae44755Sopenharmony_ci * Unless required by applicable law or agreed to in writing, software
10bae44755Sopenharmony_ci * distributed under the License is distributed on an "AS IS" BASIS,
11bae44755Sopenharmony_ci * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12bae44755Sopenharmony_ci * See the License for the specific language governing permissions and
13bae44755Sopenharmony_ci * limitations under the License.
14bae44755Sopenharmony_ci */
15bae44755Sopenharmony_ci
16bae44755Sopenharmony_ci#include <stdint.h>
17bae44755Sopenharmony_ci#include <stdio.h>
18bae44755Sopenharmony_ci#include <stdbool.h>
19bae44755Sopenharmony_ci#include <stdlib.h>
20bae44755Sopenharmony_ci#include <errno.h>
21bae44755Sopenharmony_ci#include <string.h>
22bae44755Sopenharmony_ci#include <limits.h>
23bae44755Sopenharmony_ci#include <sys/stat.h>
24bae44755Sopenharmony_ci#include "securec.h"
25bae44755Sopenharmony_ci#include "endian_internal.h"
26bae44755Sopenharmony_ci#include "cJSON.h"
27bae44755Sopenharmony_ci#include "create_pcid.h"
28bae44755Sopenharmony_ci#include "syscap_tool.h"
29bae44755Sopenharmony_ci#include "context_tool.h"
30bae44755Sopenharmony_ci#include "common_method.h"
31bae44755Sopenharmony_ci
32bae44755Sopenharmony_ci#ifdef SYSCAP_DEFINE_EXTERN_ENABLE
33bae44755Sopenharmony_ci#include "syscap_define_custom.h"
34bae44755Sopenharmony_ci#else
35bae44755Sopenharmony_ci#include "syscap_define.h"
36bae44755Sopenharmony_ci#endif
37bae44755Sopenharmony_ci
38bae44755Sopenharmony_ci#define SYSCAP_PREFIX_LEN 17
39bae44755Sopenharmony_ci#define SINGLE_FEAT_LEN (SINGLE_SYSCAP_LEN - SYSCAP_PREFIX_LEN)
40bae44755Sopenharmony_ci#define DOUBLE_SPACE 2
41bae44755Sopenharmony_ci#define UINT8_BIT 8
42bae44755Sopenharmony_ci#define INT_BIT 32
43bae44755Sopenharmony_ci#define RPCID_OUT_BUFFER 32
44bae44755Sopenharmony_ci#define PCID_OUT_BUFFER RPCID_OUT_BUFFER
45bae44755Sopenharmony_ci#define BYTES_OF_OS_SYSCAP 120
46bae44755Sopenharmony_ci#define U32_TO_STR_MAX_LEN 11
47bae44755Sopenharmony_ci#define STRING_FORMAT_LEN_MAX 1024
48bae44755Sopenharmony_ci
49bae44755Sopenharmony_ci#define FREE_CONVERT_OUT_RPCID_ENCODE 1
50bae44755Sopenharmony_ci#define FREE_CONTEXT_OUT_RPCID_ENCODE 2
51bae44755Sopenharmony_ci
52bae44755Sopenharmony_ci#define FREE_OUTBUFFER_AFTER_RPCIDSC 1
53bae44755Sopenharmony_ci#define FREE_MALLOC_RPISYSCAP_AFTER_RPCIDSC 2
54bae44755Sopenharmony_ci#define FREE_MALLOC_OSSYSCAP_AFTER_RPCIDSC 3
55bae44755Sopenharmony_ci#define FREE_WHOLE_SYSCAP_AFTER_RPCIDSC 4
56bae44755Sopenharmony_ci#define FREE_RPCID_ROOT_AFTER_RPCIDSC 5
57bae44755Sopenharmony_ci#define FREE_CONTEXT_OUT_AFTER_RPCIDSC 6
58bae44755Sopenharmony_ci
59bae44755Sopenharmony_cistruct JsonObjectSysCap {
60bae44755Sopenharmony_ci    cJSON *cjsonObjectRoot;
61bae44755Sopenharmony_ci    cJSON *sysCapPtr;
62bae44755Sopenharmony_ci};
63bae44755Sopenharmony_ci
64bae44755Sopenharmony_cistruct FreeAfterEncodeRpcidscInfo {
65bae44755Sopenharmony_ci    char *outBuffer;
66bae44755Sopenharmony_ci    char *priSyscapArray;
67bae44755Sopenharmony_ci    uint16_t *osSysCapIndex;
68bae44755Sopenharmony_ci    cJSON *sysCapDefine;
69bae44755Sopenharmony_ci    cJSON *rpcidRoot;
70bae44755Sopenharmony_ci    char *contextBuffer;
71bae44755Sopenharmony_ci    int32_t type;
72bae44755Sopenharmony_ci    int16_t flag;
73bae44755Sopenharmony_ci};
74bae44755Sopenharmony_ci
75bae44755Sopenharmony_cistatic void FreePointerMemory(char *pcidContent, char *rpcidContent, char *pcidPriSyscap, char *rpcidPriSyscap)
76bae44755Sopenharmony_ci{
77bae44755Sopenharmony_ci    SafeFree(pcidContent);
78bae44755Sopenharmony_ci    SafeFree(rpcidPriSyscap);
79bae44755Sopenharmony_ci    SafeFree(rpcidContent);
80bae44755Sopenharmony_ci    SafeFree(pcidPriSyscap);
81bae44755Sopenharmony_ci}
82bae44755Sopenharmony_ci
83bae44755Sopenharmony_cistatic int32_t FillOsCapLength(char *convertedBuffer, char *contextBuffer, struct JsonObjectSysCap gJsonObjectSysCap,
84bae44755Sopenharmony_ci        uint32_t sysCapSize, int32_t ret)
85bae44755Sopenharmony_ci{
86bae44755Sopenharmony_ci    RPCIDHead *headPtr = NULL;
87bae44755Sopenharmony_ci    char *fillTmpPtr = NULL;
88bae44755Sopenharmony_ci    cJSON *apiVerItem = NULL;
89bae44755Sopenharmony_ci    cJSON *arrayItemPtr = NULL;
90bae44755Sopenharmony_ci
91bae44755Sopenharmony_ci    headPtr = (RPCIDHead *)convertedBuffer;
92bae44755Sopenharmony_ci    apiVerItem = cJSON_GetObjectItem(gJsonObjectSysCap.cjsonObjectRoot, "api_version");
93bae44755Sopenharmony_ci    if (apiVerItem == NULL || !cJSON_IsNumber(apiVerItem)) {
94bae44755Sopenharmony_ci        PRINT_ERR("get \"api_version\" failed\n");
95bae44755Sopenharmony_ci        return -1;
96bae44755Sopenharmony_ci    }
97bae44755Sopenharmony_ci    headPtr->apiVersion = HtonsInter((uint16_t)apiVerItem->valueint);
98bae44755Sopenharmony_ci    headPtr->apiVersionType = 1;
99bae44755Sopenharmony_ci
100bae44755Sopenharmony_ci    fillTmpPtr = convertedBuffer + sizeof(RPCIDHead);
101bae44755Sopenharmony_ci    *(uint16_t *)fillTmpPtr = HtonsInter(2); // 2, SysCap Type, 2: request Cap
102bae44755Sopenharmony_ci    fillTmpPtr += sizeof(uint16_t);
103bae44755Sopenharmony_ci    // fill osCap Length
104bae44755Sopenharmony_ci    *(uint16_t *)fillTmpPtr = HtonsInter((uint16_t)(sysCapSize * SINGLE_FEAT_LEN));
105bae44755Sopenharmony_ci    fillTmpPtr += sizeof(uint16_t);
106bae44755Sopenharmony_ci    for (uint32_t i = 0; i < sysCapSize; i++) {
107bae44755Sopenharmony_ci        arrayItemPtr = cJSON_GetArrayItem(gJsonObjectSysCap.sysCapPtr, (int)i);
108bae44755Sopenharmony_ci        if (arrayItemPtr->valuestring == NULL) {
109bae44755Sopenharmony_ci            PRINT_ERR("arrayItemPtr->valuestring is NULL\n");
110bae44755Sopenharmony_ci            return -1;
111bae44755Sopenharmony_ci        }
112bae44755Sopenharmony_ci
113bae44755Sopenharmony_ci        char *pointPos = strchr(arrayItemPtr->valuestring, '.');
114bae44755Sopenharmony_ci        if (pointPos == NULL) {
115bae44755Sopenharmony_ci            PRINT_ERR("context of \"syscap\" array is invalid\n");
116bae44755Sopenharmony_ci            return -1;
117bae44755Sopenharmony_ci        }
118bae44755Sopenharmony_ci        ret = strncmp(arrayItemPtr->valuestring, "SystemCapability.", pointPos - arrayItemPtr->valuestring + 1);
119bae44755Sopenharmony_ci        if (ret != 0) {
120bae44755Sopenharmony_ci            PRINT_ERR("context of \"syscap\" array is invalid\n");
121bae44755Sopenharmony_ci            return -1;
122bae44755Sopenharmony_ci        }
123bae44755Sopenharmony_ci
124bae44755Sopenharmony_ci        ret = memcpy_s(fillTmpPtr, SINGLE_FEAT_LEN, pointPos + 1, strlen(pointPos + 1));
125bae44755Sopenharmony_ci        if (ret != 0) {
126bae44755Sopenharmony_ci            PRINT_ERR("context of \"syscap\" array is invalid\n");
127bae44755Sopenharmony_ci            return -1;
128bae44755Sopenharmony_ci        }
129bae44755Sopenharmony_ci        fillTmpPtr += SINGLE_FEAT_LEN;
130bae44755Sopenharmony_ci    }
131bae44755Sopenharmony_ci    return ret;
132bae44755Sopenharmony_ci}
133bae44755Sopenharmony_ci
134bae44755Sopenharmony_cistatic int32_t FreeAfterRPCIDEncode(
135bae44755Sopenharmony_ci    cJSON *cjsonObjectRoot, char *convertedBuffer, char *contextBuffer, int32_t type, int32_t ret)
136bae44755Sopenharmony_ci{
137bae44755Sopenharmony_ci    if (type == FREE_CONVERT_OUT_RPCID_ENCODE) {
138bae44755Sopenharmony_ci        free(convertedBuffer);
139bae44755Sopenharmony_ci    }
140bae44755Sopenharmony_ci    cJSON_Delete(cjsonObjectRoot);
141bae44755Sopenharmony_ci    FreeContextBuffer(contextBuffer);
142bae44755Sopenharmony_ci    return ret;
143bae44755Sopenharmony_ci}
144bae44755Sopenharmony_ci
145bae44755Sopenharmony_ciint32_t RPCIDEncode(char *inputFile, char *outputPath)
146bae44755Sopenharmony_ci{
147bae44755Sopenharmony_ci    char *contextBuffer = NULL;
148bae44755Sopenharmony_ci    uint32_t bufferLen, sysCapSize;
149bae44755Sopenharmony_ci    struct JsonObjectSysCap gJsonObjectSysCap;
150bae44755Sopenharmony_ci    gJsonObjectSysCap.cjsonObjectRoot = NULL;
151bae44755Sopenharmony_ci    gJsonObjectSysCap.sysCapPtr = NULL;
152bae44755Sopenharmony_ci
153bae44755Sopenharmony_ci    if (CheckFileAndGetFileContext(inputFile, &contextBuffer, &bufferLen) != 0) {
154bae44755Sopenharmony_ci        return -1;
155bae44755Sopenharmony_ci    }
156bae44755Sopenharmony_ci
157bae44755Sopenharmony_ci    gJsonObjectSysCap.cjsonObjectRoot = cJSON_ParseWithLength(contextBuffer, bufferLen);
158bae44755Sopenharmony_ci    if (gJsonObjectSysCap.cjsonObjectRoot == NULL) {
159bae44755Sopenharmony_ci        PRINT_ERR("cJSON_Parse failed, context buffer is:\n%s\n", contextBuffer);
160bae44755Sopenharmony_ci        return FreeAfterRPCIDEncode(gJsonObjectSysCap.cjsonObjectRoot, NULL, contextBuffer,
161bae44755Sopenharmony_ci                                    FREE_CONTEXT_OUT_RPCID_ENCODE, -1);
162bae44755Sopenharmony_ci    }
163bae44755Sopenharmony_ci
164bae44755Sopenharmony_ci    gJsonObjectSysCap.sysCapPtr = cJSON_GetObjectItem(gJsonObjectSysCap.cjsonObjectRoot, "syscap");
165bae44755Sopenharmony_ci    if (gJsonObjectSysCap.sysCapPtr == NULL || !cJSON_IsArray(gJsonObjectSysCap.sysCapPtr)) {
166bae44755Sopenharmony_ci        PRINT_ERR("get \"syscap\" object failed.\n");
167bae44755Sopenharmony_ci        return FreeAfterRPCIDEncode(gJsonObjectSysCap.cjsonObjectRoot, NULL, contextBuffer,
168bae44755Sopenharmony_ci                                    FREE_CONTEXT_OUT_RPCID_ENCODE, -1);
169bae44755Sopenharmony_ci    }
170bae44755Sopenharmony_ci
171bae44755Sopenharmony_ci    int32_t ret = cJSON_GetArraySize(gJsonObjectSysCap.sysCapPtr);
172bae44755Sopenharmony_ci    if (ret < 0) {
173bae44755Sopenharmony_ci        PRINT_ERR("get \"syscap\" array size failed\n");
174bae44755Sopenharmony_ci        return FreeAfterRPCIDEncode(gJsonObjectSysCap.cjsonObjectRoot, NULL, contextBuffer,
175bae44755Sopenharmony_ci                                    FREE_CONTEXT_OUT_RPCID_ENCODE, -1);
176bae44755Sopenharmony_ci    }
177bae44755Sopenharmony_ci
178bae44755Sopenharmony_ci    // 2, to save SysCaptype & SysCapLength
179bae44755Sopenharmony_ci    sysCapSize = (uint32_t) ret;
180bae44755Sopenharmony_ci    uint32_t convertedBufLen = sizeof(RPCIDHead);
181bae44755Sopenharmony_ci    convertedBufLen += (DOUBLE_SPACE * sizeof(uint16_t) + sysCapSize * SINGLE_FEAT_LEN);
182bae44755Sopenharmony_ci    char *convertedBuffer = (char *) malloc(convertedBufLen);
183bae44755Sopenharmony_ci    if (convertedBuffer == NULL) {
184bae44755Sopenharmony_ci        PRINT_ERR("malloc failed\n");
185bae44755Sopenharmony_ci        return FreeAfterRPCIDEncode(gJsonObjectSysCap.cjsonObjectRoot, convertedBuffer, contextBuffer,
186bae44755Sopenharmony_ci                                    FREE_CONTEXT_OUT_RPCID_ENCODE, -1);
187bae44755Sopenharmony_ci    }
188bae44755Sopenharmony_ci    (void) memset_s(convertedBuffer, convertedBufLen, 0, convertedBufLen);
189bae44755Sopenharmony_ci
190bae44755Sopenharmony_ci    if (FillOsCapLength(convertedBuffer, contextBuffer, gJsonObjectSysCap, sysCapSize, ret) == -1) {
191bae44755Sopenharmony_ci        return FreeAfterRPCIDEncode(gJsonObjectSysCap.cjsonObjectRoot, convertedBuffer, contextBuffer,
192bae44755Sopenharmony_ci                                    FREE_CONVERT_OUT_RPCID_ENCODE, -1);
193bae44755Sopenharmony_ci    }
194bae44755Sopenharmony_ci    ret = ConvertedContextSaveAsFile(outputPath, "rpcid.sc", convertedBuffer, convertedBufLen);
195bae44755Sopenharmony_ci    if (ret != 0) {
196bae44755Sopenharmony_ci        PRINT_ERR("ConvertedContextSaveAsFile failed, outputPath:%s, filename:rpcid.sc\n", outputPath);
197bae44755Sopenharmony_ci    }
198bae44755Sopenharmony_ci    return FreeAfterRPCIDEncode(gJsonObjectSysCap.cjsonObjectRoot, convertedBuffer, contextBuffer,
199bae44755Sopenharmony_ci                                FREE_CONVERT_OUT_RPCID_ENCODE, ret);
200bae44755Sopenharmony_ci}
201bae44755Sopenharmony_ci
202bae44755Sopenharmony_cistatic int32_t ParseRpcidToJson(char *input, uint32_t inputLen, cJSON *rpcidJson)
203bae44755Sopenharmony_ci{
204bae44755Sopenharmony_ci    uint32_t i;
205bae44755Sopenharmony_ci    int32_t ret = 0;
206bae44755Sopenharmony_ci    uint16_t sysCapLength = NtohsInter(*(uint16_t *)(input + sizeof(uint32_t)));
207bae44755Sopenharmony_ci    if (sysCapLength > inputLen - sizeof(uint32_t)) {
208bae44755Sopenharmony_ci        PRINT_ERR("Get sysCapLength(%u) error, inputLen = %u\n", sysCapLength, inputLen);
209bae44755Sopenharmony_ci        return -1;
210bae44755Sopenharmony_ci    }
211bae44755Sopenharmony_ci    uint16_t sysCapCount = sysCapLength / SINGLE_FEAT_LEN;
212bae44755Sopenharmony_ci    char *sysCapBegin = input + sizeof(RPCIDHead) + sizeof(uint32_t);
213bae44755Sopenharmony_ci    RPCIDHead *rpcidHeader = (RPCIDHead *)input;
214bae44755Sopenharmony_ci    cJSON *sysCapJson = cJSON_CreateArray();
215bae44755Sopenharmony_ci    if (sysCapJson == NULL) {
216bae44755Sopenharmony_ci        PRINT_ERR("Failed to create sysCapJson array\n");
217bae44755Sopenharmony_ci        return -1;
218bae44755Sopenharmony_ci    }
219bae44755Sopenharmony_ci
220bae44755Sopenharmony_ci    for (i = 0; i < sysCapCount; i++) {
221bae44755Sopenharmony_ci        char *temp = sysCapBegin + i * SINGLE_FEAT_LEN;
222bae44755Sopenharmony_ci        if (strlen(temp) >= SINGLE_FEAT_LEN || strlen(temp) == 0) {
223bae44755Sopenharmony_ci            PRINT_ERR("Get SysCap failed, string length(%u) error.\n", (uint32_t)strlen(temp));
224bae44755Sopenharmony_ci            cJSON_Delete(sysCapJson);
225bae44755Sopenharmony_ci            return -1;
226bae44755Sopenharmony_ci        }
227bae44755Sopenharmony_ci        char buffer[SINGLE_SYSCAP_LEN] = "SystemCapability.";
228bae44755Sopenharmony_ci
229bae44755Sopenharmony_ci        ret = strncat_s(buffer, sizeof(buffer), temp, SINGLE_FEAT_LEN);
230bae44755Sopenharmony_ci        if (ret != EOK) {
231bae44755Sopenharmony_ci            PRINT_ERR("strncat_s failed.\n");
232bae44755Sopenharmony_ci            cJSON_Delete(sysCapJson);
233bae44755Sopenharmony_ci            return ret;
234bae44755Sopenharmony_ci        }
235bae44755Sopenharmony_ci
236bae44755Sopenharmony_ci        if (!cJSON_AddItemToArray(sysCapJson, cJSON_CreateString(buffer))) {
237bae44755Sopenharmony_ci            PRINT_ERR("Add syscap string to json failed.\n");
238bae44755Sopenharmony_ci            cJSON_Delete(sysCapJson);
239bae44755Sopenharmony_ci            return -1;
240bae44755Sopenharmony_ci        }
241bae44755Sopenharmony_ci    }
242bae44755Sopenharmony_ci
243bae44755Sopenharmony_ci    if (!cJSON_AddNumberToObject(rpcidJson, "api_version", NtohsInter(rpcidHeader->apiVersion))) {
244bae44755Sopenharmony_ci        PRINT_ERR("Add api_version to json failed.\n");
245bae44755Sopenharmony_ci        cJSON_Delete(sysCapJson);
246bae44755Sopenharmony_ci        return -1;
247bae44755Sopenharmony_ci    }
248bae44755Sopenharmony_ci    if (!cJSON_AddItemToObject(rpcidJson, "syscap", sysCapJson)) {
249bae44755Sopenharmony_ci        PRINT_ERR("Add syscap to json failed.\n");
250bae44755Sopenharmony_ci        cJSON_Delete(sysCapJson);
251bae44755Sopenharmony_ci        return -1;
252bae44755Sopenharmony_ci    }
253bae44755Sopenharmony_ci
254bae44755Sopenharmony_ci    return ret;
255bae44755Sopenharmony_ci}
256bae44755Sopenharmony_ci
257bae44755Sopenharmony_ciint32_t RPCIDDecode(char *inputFile, char *outputPath)
258bae44755Sopenharmony_ci{
259bae44755Sopenharmony_ci    int32_t ret = 0;
260bae44755Sopenharmony_ci    char *contextBuffer = NULL;
261bae44755Sopenharmony_ci    char *convertedBuffer = NULL;
262bae44755Sopenharmony_ci    uint32_t bufferLen;
263bae44755Sopenharmony_ci
264bae44755Sopenharmony_ci    if (inputFile == NULL) {
265bae44755Sopenharmony_ci        PRINT_ERR("inputFile is null.\n");
266bae44755Sopenharmony_ci        return -1;
267bae44755Sopenharmony_ci    }
268bae44755Sopenharmony_ci
269bae44755Sopenharmony_ci    // check rpcid.sc
270bae44755Sopenharmony_ci    if (CheckRpcidFormat(inputFile, &contextBuffer, &bufferLen)) {
271bae44755Sopenharmony_ci        PRINT_ERR("Check rpcid.sc format failed. Input failed: %s\n", inputFile);
272bae44755Sopenharmony_ci        goto FREE_CONTEXT_OUT;
273bae44755Sopenharmony_ci    }
274bae44755Sopenharmony_ci
275bae44755Sopenharmony_ci    // parse rpcid to json
276bae44755Sopenharmony_ci    cJSON *rpcidRoot = cJSON_CreateObject();
277bae44755Sopenharmony_ci    if (ParseRpcidToJson(contextBuffer, bufferLen, rpcidRoot) != 0) {
278bae44755Sopenharmony_ci        PRINT_ERR("Prase rpcid to json failed. Input failed: %s\n", inputFile);
279bae44755Sopenharmony_ci        goto FREE_RPCID_ROOT;
280bae44755Sopenharmony_ci    }
281bae44755Sopenharmony_ci
282bae44755Sopenharmony_ci    // save to json file
283bae44755Sopenharmony_ci    convertedBuffer = cJSON_Print(rpcidRoot);
284bae44755Sopenharmony_ci    if (convertedBuffer == NULL) {
285bae44755Sopenharmony_ci        PRINT_ERR("cJSON_Print failed to create JSON string\n");
286bae44755Sopenharmony_ci        goto FREE_RPCID_ROOT;
287bae44755Sopenharmony_ci    }
288bae44755Sopenharmony_ci
289bae44755Sopenharmony_ci    ret = ConvertedContextSaveAsFile(outputPath, "rpcid.json", convertedBuffer, strlen(convertedBuffer));
290bae44755Sopenharmony_ci    if (ret != 0) {
291bae44755Sopenharmony_ci        PRINT_ERR("ConvertedContextSaveAsFile failed, outputPath:%s, filename:rpcid.json\n", outputPath);
292bae44755Sopenharmony_ci        cJSON_free(convertedBuffer);
293bae44755Sopenharmony_ci        goto FREE_RPCID_ROOT;
294bae44755Sopenharmony_ci    }
295bae44755Sopenharmony_ci    cJSON_free(convertedBuffer);
296bae44755Sopenharmony_ciFREE_RPCID_ROOT:
297bae44755Sopenharmony_ci    cJSON_Delete(rpcidRoot);
298bae44755Sopenharmony_ciFREE_CONTEXT_OUT:
299bae44755Sopenharmony_ci    FreeContextBuffer(contextBuffer);
300bae44755Sopenharmony_ci    return ret;
301bae44755Sopenharmony_ci}
302bae44755Sopenharmony_ci
303bae44755Sopenharmony_cistatic int SetOsSysCapBitMap(uint8_t *out, uint16_t outLen, const uint16_t *index, uint16_t indexLen)
304bae44755Sopenharmony_ci{
305bae44755Sopenharmony_ci    uint16_t sector, pos;
306bae44755Sopenharmony_ci
307bae44755Sopenharmony_ci    if (outLen != BYTES_OF_OS_SYSCAP) {
308bae44755Sopenharmony_ci        PRINT_ERR("Input array error.\n");
309bae44755Sopenharmony_ci        return -1;
310bae44755Sopenharmony_ci    }
311bae44755Sopenharmony_ci
312bae44755Sopenharmony_ci    for (uint16_t i = 0; i < indexLen; i++) {
313bae44755Sopenharmony_ci        sector = index[i] / UINT8_BIT;
314bae44755Sopenharmony_ci        pos = index[i] % UINT8_BIT;
315bae44755Sopenharmony_ci        if (sector >= BYTES_OF_OS_SYSCAP) {
316bae44755Sopenharmony_ci            PRINT_ERR("Syscap num(%u) out of range(120).\n", sector);
317bae44755Sopenharmony_ci            return -1;
318bae44755Sopenharmony_ci        }
319bae44755Sopenharmony_ci        out[sector] |=  (1 << pos);
320bae44755Sopenharmony_ci    }
321bae44755Sopenharmony_ci    return 0;
322bae44755Sopenharmony_ci}
323bae44755Sopenharmony_ci
324bae44755Sopenharmony_cistatic int32_t PrintOutputToFile(struct FreeAfterEncodeRpcidscInfo freeAfterEncodeRpcidscInfo,
325bae44755Sopenharmony_ci        uint32_t outUint[RPCID_OUT_BUFFER], uint16_t indexPri, char *outDirPath)
326bae44755Sopenharmony_ci{
327bae44755Sopenharmony_ci    int32_t ret = 0;
328bae44755Sopenharmony_ci
329bae44755Sopenharmony_ci    uint16_t outBufferLen = U32_TO_STR_MAX_LEN * RPCID_OUT_BUFFER + SINGLE_SYSCAP_LEN * indexPri;
330bae44755Sopenharmony_ci    freeAfterEncodeRpcidscInfo.outBuffer = (char *)malloc(outBufferLen);
331bae44755Sopenharmony_ci    if (freeAfterEncodeRpcidscInfo.outBuffer == NULL) {
332bae44755Sopenharmony_ci        PRINT_ERR("malloc(%u) failed.\n", outBufferLen);
333bae44755Sopenharmony_ci        freeAfterEncodeRpcidscInfo.flag = 1;
334bae44755Sopenharmony_ci        return ret;
335bae44755Sopenharmony_ci    }
336bae44755Sopenharmony_ci
337bae44755Sopenharmony_ci    freeAfterEncodeRpcidscInfo.type = FREE_OUTBUFFER_AFTER_RPCIDSC;
338bae44755Sopenharmony_ci    (void)memset_s(freeAfterEncodeRpcidscInfo.outBuffer, outBufferLen, 0, outBufferLen);
339bae44755Sopenharmony_ci    ret = sprintf_s(freeAfterEncodeRpcidscInfo.outBuffer, outBufferLen, "%u", outUint[0]);
340bae44755Sopenharmony_ci    if (ret == -1) {
341bae44755Sopenharmony_ci        PRINT_ERR("sprintf_s failed.\n");
342bae44755Sopenharmony_ci        freeAfterEncodeRpcidscInfo.flag = 1;
343bae44755Sopenharmony_ci        return ret;
344bae44755Sopenharmony_ci    }
345bae44755Sopenharmony_ci    for (int i = 1; i < RPCID_OUT_BUFFER; i++) {
346bae44755Sopenharmony_ci        ret = sprintf_s(freeAfterEncodeRpcidscInfo.outBuffer, outBufferLen, "%s,%u",
347bae44755Sopenharmony_ci                freeAfterEncodeRpcidscInfo.outBuffer, outUint[i]);
348bae44755Sopenharmony_ci        if (ret == -1) {
349bae44755Sopenharmony_ci            PRINT_ERR("sprintf_s failed.\n");
350bae44755Sopenharmony_ci            freeAfterEncodeRpcidscInfo.flag = 1;
351bae44755Sopenharmony_ci            return ret;
352bae44755Sopenharmony_ci        }
353bae44755Sopenharmony_ci    }
354bae44755Sopenharmony_ci
355bae44755Sopenharmony_ci    for (uint16_t i = 0; i < indexPri; i++) {
356bae44755Sopenharmony_ci        ret = sprintf_s(freeAfterEncodeRpcidscInfo.outBuffer, outBufferLen, "%s,%s",
357bae44755Sopenharmony_ci            freeAfterEncodeRpcidscInfo.outBuffer, freeAfterEncodeRpcidscInfo.priSyscapArray + i * SINGLE_SYSCAP_LEN);
358bae44755Sopenharmony_ci        if (ret == -1) {
359bae44755Sopenharmony_ci            PRINT_ERR("sprintf_s failed.\n");
360bae44755Sopenharmony_ci            freeAfterEncodeRpcidscInfo.flag = 1;
361bae44755Sopenharmony_ci            return ret;
362bae44755Sopenharmony_ci        }
363bae44755Sopenharmony_ci    }
364bae44755Sopenharmony_ci
365bae44755Sopenharmony_ci    const char outputFilename[] = "rpcid.txt";
366bae44755Sopenharmony_ci    ret = ConvertedContextSaveAsFile(outDirPath, outputFilename, freeAfterEncodeRpcidscInfo.outBuffer,
367bae44755Sopenharmony_ci            strlen(freeAfterEncodeRpcidscInfo.outBuffer));
368bae44755Sopenharmony_ci    if (ret != 0) {
369bae44755Sopenharmony_ci        PRINT_ERR("Save to txt file failed. Output path:%s/%s\n", outDirPath, outputFilename);
370bae44755Sopenharmony_ci        freeAfterEncodeRpcidscInfo.flag = 1;
371bae44755Sopenharmony_ci    }
372bae44755Sopenharmony_ci    return ret;
373bae44755Sopenharmony_ci}
374bae44755Sopenharmony_ci
375bae44755Sopenharmony_cistatic int32_t OutputSetMemAndPrintToFile(struct FreeAfterEncodeRpcidscInfo freeAfterEncodeRpcidscInfo,
376bae44755Sopenharmony_ci    int32_t sysCapArraySize, cJSON *sysCapArray, char *outDirPath)
377bae44755Sopenharmony_ci{
378bae44755Sopenharmony_ci    char *priSyscap = NULL;
379bae44755Sopenharmony_ci    cJSON *cJsonTemp = NULL;
380bae44755Sopenharmony_ci    uint16_t indexPri = 0;
381bae44755Sopenharmony_ci    int32_t ret = 0;
382bae44755Sopenharmony_ci    freeAfterEncodeRpcidscInfo.type = FREE_MALLOC_OSSYSCAP_AFTER_RPCIDSC;
383bae44755Sopenharmony_ci
384bae44755Sopenharmony_ci    (void)memset_s(freeAfterEncodeRpcidscInfo.osSysCapIndex, sizeof(uint16_t) * sysCapArraySize,
385bae44755Sopenharmony_ci                   0, sizeof(uint16_t) * sysCapArraySize);
386bae44755Sopenharmony_ci    // malloc for save private syscap string
387bae44755Sopenharmony_ci    freeAfterEncodeRpcidscInfo.priSyscapArray = (char *)malloc((uint32_t)sysCapArraySize * SINGLE_SYSCAP_LEN);
388bae44755Sopenharmony_ci    if (freeAfterEncodeRpcidscInfo.priSyscapArray == NULL) {
389bae44755Sopenharmony_ci        PRINT_ERR("malloc(%d) failed.\n", sysCapArraySize * SINGLE_SYSCAP_LEN);
390bae44755Sopenharmony_ci        freeAfterEncodeRpcidscInfo.flag = 1;
391bae44755Sopenharmony_ci        return ret;
392bae44755Sopenharmony_ci    }
393bae44755Sopenharmony_ci    (void)memset_s(freeAfterEncodeRpcidscInfo.priSyscapArray, (size_t)(sysCapArraySize * SINGLE_SYSCAP_LEN),
394bae44755Sopenharmony_ci                   0, (size_t)(sysCapArraySize * SINGLE_SYSCAP_LEN));
395bae44755Sopenharmony_ci    priSyscap = freeAfterEncodeRpcidscInfo.priSyscapArray;
396bae44755Sopenharmony_ci    // part os syscap and ptivate syscap
397bae44755Sopenharmony_ci    uint16_t indexOs = 0;
398bae44755Sopenharmony_ci    for (int i = 0; i < sysCapArraySize; i++) {
399bae44755Sopenharmony_ci        cJSON *cJsonItem = cJSON_GetArrayItem(sysCapArray, i);
400bae44755Sopenharmony_ci        if (cJsonItem->valuestring == NULL) {
401bae44755Sopenharmony_ci            continue;
402bae44755Sopenharmony_ci        }
403bae44755Sopenharmony_ci        cJsonTemp = cJSON_GetObjectItem(freeAfterEncodeRpcidscInfo.sysCapDefine, cJsonItem->valuestring);
404bae44755Sopenharmony_ci        if (cJsonTemp != NULL && cJSON_IsNumber(cJsonTemp)) {
405bae44755Sopenharmony_ci            freeAfterEncodeRpcidscInfo.osSysCapIndex[indexOs++] = (uint16_t)(cJsonTemp->valueint);
406bae44755Sopenharmony_ci        } else {
407bae44755Sopenharmony_ci            ret = strcpy_s(priSyscap, SINGLE_SYSCAP_LEN, cJsonItem->valuestring);
408bae44755Sopenharmony_ci            if (ret != EOK) {
409bae44755Sopenharmony_ci                PRINT_ERR("strcpy_s failed.\n");
410bae44755Sopenharmony_ci                freeAfterEncodeRpcidscInfo.flag = 1;
411bae44755Sopenharmony_ci                return ret;
412bae44755Sopenharmony_ci            }
413bae44755Sopenharmony_ci            priSyscap += SINGLE_SYSCAP_LEN;
414bae44755Sopenharmony_ci            indexPri++;
415bae44755Sopenharmony_ci        }
416bae44755Sopenharmony_ci    }
417bae44755Sopenharmony_ci    uint32_t outUint[RPCID_OUT_BUFFER] = {0};
418bae44755Sopenharmony_ci    outUint[0] = *(uint32_t *)freeAfterEncodeRpcidscInfo.contextBuffer;
419bae44755Sopenharmony_ci    outUint[1] = *(uint32_t *)(freeAfterEncodeRpcidscInfo.contextBuffer + sizeof(uint32_t));
420bae44755Sopenharmony_ci    uint8_t *osOutUint = (uint8_t *)(outUint + 2);
421bae44755Sopenharmony_ci    // 120, len of osOutUint
422bae44755Sopenharmony_ci    if (SetOsSysCapBitMap(osOutUint, 120, freeAfterEncodeRpcidscInfo.osSysCapIndex, indexOs) != 0) {
423bae44755Sopenharmony_ci        PRINT_ERR("Set os syscap bit map failed.\n");
424bae44755Sopenharmony_ci        freeAfterEncodeRpcidscInfo.flag = 1;
425bae44755Sopenharmony_ci        return ret;
426bae44755Sopenharmony_ci    }
427bae44755Sopenharmony_ci    return PrintOutputToFile(freeAfterEncodeRpcidscInfo, outUint, indexPri, outDirPath);
428bae44755Sopenharmony_ci}
429bae44755Sopenharmony_ci
430bae44755Sopenharmony_cistatic int32_t FreeAfterEncodeRpcidsc(struct FreeAfterEncodeRpcidscInfo freeAfterEncodeRpcidscInfo, int32_t type,
431bae44755Sopenharmony_ci        int32_t ret)
432bae44755Sopenharmony_ci{
433bae44755Sopenharmony_ci    switch (type) {
434bae44755Sopenharmony_ci        case FREE_OUTBUFFER_AFTER_RPCIDSC:
435bae44755Sopenharmony_ci            free(freeAfterEncodeRpcidscInfo.outBuffer);
436bae44755Sopenharmony_ci            free(freeAfterEncodeRpcidscInfo.priSyscapArray);
437bae44755Sopenharmony_ci            free(freeAfterEncodeRpcidscInfo.osSysCapIndex);
438bae44755Sopenharmony_ci            cJSON_Delete(freeAfterEncodeRpcidscInfo.sysCapDefine);
439bae44755Sopenharmony_ci            cJSON_Delete(freeAfterEncodeRpcidscInfo.rpcidRoot);
440bae44755Sopenharmony_ci            FreeContextBuffer(freeAfterEncodeRpcidscInfo.contextBuffer);
441bae44755Sopenharmony_ci            break;
442bae44755Sopenharmony_ci        case FREE_MALLOC_RPISYSCAP_AFTER_RPCIDSC:
443bae44755Sopenharmony_ci            free(freeAfterEncodeRpcidscInfo.priSyscapArray);
444bae44755Sopenharmony_ci            free(freeAfterEncodeRpcidscInfo.osSysCapIndex);
445bae44755Sopenharmony_ci            cJSON_Delete(freeAfterEncodeRpcidscInfo.sysCapDefine);
446bae44755Sopenharmony_ci            cJSON_Delete(freeAfterEncodeRpcidscInfo.rpcidRoot);
447bae44755Sopenharmony_ci            FreeContextBuffer(freeAfterEncodeRpcidscInfo.contextBuffer);
448bae44755Sopenharmony_ci            break;
449bae44755Sopenharmony_ci        case FREE_MALLOC_OSSYSCAP_AFTER_RPCIDSC:
450bae44755Sopenharmony_ci            free(freeAfterEncodeRpcidscInfo.osSysCapIndex);
451bae44755Sopenharmony_ci            cJSON_Delete(freeAfterEncodeRpcidscInfo.sysCapDefine);
452bae44755Sopenharmony_ci            cJSON_Delete(freeAfterEncodeRpcidscInfo.rpcidRoot);
453bae44755Sopenharmony_ci            FreeContextBuffer(freeAfterEncodeRpcidscInfo.contextBuffer);
454bae44755Sopenharmony_ci            break;
455bae44755Sopenharmony_ci        case FREE_WHOLE_SYSCAP_AFTER_RPCIDSC:
456bae44755Sopenharmony_ci            cJSON_Delete(freeAfterEncodeRpcidscInfo.sysCapDefine);
457bae44755Sopenharmony_ci            cJSON_Delete(freeAfterEncodeRpcidscInfo.rpcidRoot);
458bae44755Sopenharmony_ci            FreeContextBuffer(freeAfterEncodeRpcidscInfo.contextBuffer);
459bae44755Sopenharmony_ci            break;
460bae44755Sopenharmony_ci        case FREE_RPCID_ROOT_AFTER_RPCIDSC:
461bae44755Sopenharmony_ci            cJSON_Delete(freeAfterEncodeRpcidscInfo.rpcidRoot);
462bae44755Sopenharmony_ci            FreeContextBuffer(freeAfterEncodeRpcidscInfo.contextBuffer);
463bae44755Sopenharmony_ci            break;
464bae44755Sopenharmony_ci        case FREE_CONTEXT_OUT_AFTER_RPCIDSC:
465bae44755Sopenharmony_ci        default:
466bae44755Sopenharmony_ci            FreeContextBuffer(freeAfterEncodeRpcidscInfo.contextBuffer);
467bae44755Sopenharmony_ci    }
468bae44755Sopenharmony_ci    return ret;
469bae44755Sopenharmony_ci}
470bae44755Sopenharmony_ci
471bae44755Sopenharmony_ciint32_t EncodeRpcidscToString(char *inputFile, char *outDirPath)
472bae44755Sopenharmony_ci{
473bae44755Sopenharmony_ci    int32_t ret = 0;
474bae44755Sopenharmony_ci    int32_t sysCapArraySize;
475bae44755Sopenharmony_ci    uint32_t bufferLen;
476bae44755Sopenharmony_ci    cJSON *sysCapArray = NULL;
477bae44755Sopenharmony_ci    struct FreeAfterEncodeRpcidscInfo freeAfterEncodeRpcidscInfo;
478bae44755Sopenharmony_ci    freeAfterEncodeRpcidscInfo.priSyscapArray = NULL;
479bae44755Sopenharmony_ci    freeAfterEncodeRpcidscInfo.osSysCapIndex = NULL;
480bae44755Sopenharmony_ci    freeAfterEncodeRpcidscInfo.sysCapDefine = NULL;
481bae44755Sopenharmony_ci    freeAfterEncodeRpcidscInfo.rpcidRoot = NULL;
482bae44755Sopenharmony_ci    freeAfterEncodeRpcidscInfo.outBuffer = NULL;
483bae44755Sopenharmony_ci    freeAfterEncodeRpcidscInfo.contextBuffer = NULL;
484bae44755Sopenharmony_ci    freeAfterEncodeRpcidscInfo.type = 0;
485bae44755Sopenharmony_ci    freeAfterEncodeRpcidscInfo.flag = 0;
486bae44755Sopenharmony_ci
487bae44755Sopenharmony_ci    if (inputFile == NULL) {
488bae44755Sopenharmony_ci        PRINT_ERR("inputFile is null.\n");
489bae44755Sopenharmony_ci        return -1;
490bae44755Sopenharmony_ci    }
491bae44755Sopenharmony_ci
492bae44755Sopenharmony_ci    // check rpcid.sc
493bae44755Sopenharmony_ci    if (CheckRpcidFormat(inputFile, &freeAfterEncodeRpcidscInfo.contextBuffer, &bufferLen) != 0) {
494bae44755Sopenharmony_ci        PRINT_ERR("Check rpcid.sc format failed. Input file: %s\n", inputFile);
495bae44755Sopenharmony_ci        return FreeAfterEncodeRpcidsc(freeAfterEncodeRpcidscInfo, FREE_CONTEXT_OUT_AFTER_RPCIDSC, ret);
496bae44755Sopenharmony_ci    }
497bae44755Sopenharmony_ci
498bae44755Sopenharmony_ci    // parse rpcid to json
499bae44755Sopenharmony_ci    freeAfterEncodeRpcidscInfo.rpcidRoot = cJSON_CreateObject();
500bae44755Sopenharmony_ci    if (ParseRpcidToJson(freeAfterEncodeRpcidscInfo.contextBuffer, bufferLen,
501bae44755Sopenharmony_ci            freeAfterEncodeRpcidscInfo.rpcidRoot) != 0) {
502bae44755Sopenharmony_ci        PRINT_ERR("Prase rpcid to json failed. Input file: %s\n", inputFile);
503bae44755Sopenharmony_ci        return FreeAfterEncodeRpcidsc(freeAfterEncodeRpcidscInfo, FREE_RPCID_ROOT_AFTER_RPCIDSC, ret);
504bae44755Sopenharmony_ci    }
505bae44755Sopenharmony_ci
506bae44755Sopenharmony_ci    // trans to string format
507bae44755Sopenharmony_ci    freeAfterEncodeRpcidscInfo.sysCapDefine =  CreateWholeSyscapJsonObj();
508bae44755Sopenharmony_ci    sysCapArray = cJSON_GetObjectItem(freeAfterEncodeRpcidscInfo.rpcidRoot, "syscap");
509bae44755Sopenharmony_ci    if (sysCapArray == NULL || !cJSON_IsArray(sysCapArray)) {
510bae44755Sopenharmony_ci        PRINT_ERR("Get syscap failed. Input file: %s\n", inputFile);
511bae44755Sopenharmony_ci        return FreeAfterEncodeRpcidsc(freeAfterEncodeRpcidscInfo, FREE_WHOLE_SYSCAP_AFTER_RPCIDSC, ret);
512bae44755Sopenharmony_ci    }
513bae44755Sopenharmony_ci    sysCapArraySize = cJSON_GetArraySize(sysCapArray);
514bae44755Sopenharmony_ci    if (sysCapArraySize < 0) {
515bae44755Sopenharmony_ci        PRINT_ERR("Get syscap size failed. Input file: %s\n", inputFile);
516bae44755Sopenharmony_ci        return FreeAfterEncodeRpcidsc(freeAfterEncodeRpcidscInfo, FREE_WHOLE_SYSCAP_AFTER_RPCIDSC, ret);
517bae44755Sopenharmony_ci    }
518bae44755Sopenharmony_ci    // malloc for save os syscap index
519bae44755Sopenharmony_ci    freeAfterEncodeRpcidscInfo.osSysCapIndex = (uint16_t *)malloc(sizeof(uint16_t) * sysCapArraySize);
520bae44755Sopenharmony_ci    if (freeAfterEncodeRpcidscInfo.osSysCapIndex == NULL) {
521bae44755Sopenharmony_ci        PRINT_ERR("malloc failed.\n");
522bae44755Sopenharmony_ci        return FreeAfterEncodeRpcidsc(freeAfterEncodeRpcidscInfo, FREE_WHOLE_SYSCAP_AFTER_RPCIDSC, ret);
523bae44755Sopenharmony_ci    }
524bae44755Sopenharmony_ci
525bae44755Sopenharmony_ci    ret = OutputSetMemAndPrintToFile(freeAfterEncodeRpcidscInfo, sysCapArraySize, sysCapArray, outDirPath);
526bae44755Sopenharmony_ci    if (freeAfterEncodeRpcidscInfo.flag == 1) {
527bae44755Sopenharmony_ci        return FreeAfterEncodeRpcidsc(freeAfterEncodeRpcidscInfo, freeAfterEncodeRpcidscInfo.type, ret);
528bae44755Sopenharmony_ci    }
529bae44755Sopenharmony_ci    return FreeAfterEncodeRpcidsc(freeAfterEncodeRpcidscInfo, FREE_OUTBUFFER_AFTER_RPCIDSC, ret);
530bae44755Sopenharmony_ci}
531bae44755Sopenharmony_ci
532bae44755Sopenharmony_cichar *CopyInputString(const char *inputString)
533bae44755Sopenharmony_ci{
534bae44755Sopenharmony_ci    if (inputString == NULL || *inputString == '\0') {
535bae44755Sopenharmony_ci        PRINT_ERR("inputString is null or empty.\n");
536bae44755Sopenharmony_ci        return NULL;
537bae44755Sopenharmony_ci    }
538bae44755Sopenharmony_ci    size_t inputLen = strlen(inputString);
539bae44755Sopenharmony_ci    if (inputLen > STRING_FORMAT_LEN_MAX) {
540bae44755Sopenharmony_ci        PRINT_ERR("input string too long(%zu).\n", inputLen);
541bae44755Sopenharmony_ci        return NULL;
542bae44755Sopenharmony_ci    }
543bae44755Sopenharmony_ci    char *input = (char *)malloc(inputLen + 1);
544bae44755Sopenharmony_ci    if (input == NULL) {
545bae44755Sopenharmony_ci        PRINT_ERR("malloc failed.\n");
546bae44755Sopenharmony_ci        return NULL;
547bae44755Sopenharmony_ci    }
548bae44755Sopenharmony_ci    int32_t ret = strcpy_s(input, inputLen + 1, inputString);
549bae44755Sopenharmony_ci    if (ret != EOK) {
550bae44755Sopenharmony_ci        PRINT_ERR("strcpy_s failed.\n");
551bae44755Sopenharmony_ci        free(input);
552bae44755Sopenharmony_ci        return NULL;
553bae44755Sopenharmony_ci    }
554bae44755Sopenharmony_ci    input[inputLen] = '\0';
555bae44755Sopenharmony_ci    return input;
556bae44755Sopenharmony_ci}
557bae44755Sopenharmony_ci
558bae44755Sopenharmony_ciint32_t GetPriSyscapData(char *input, char **priSyscap, uint32_t *priSyscapLen)
559bae44755Sopenharmony_ci{
560bae44755Sopenharmony_ci    // count private syscaps
561bae44755Sopenharmony_ci    uint32_t count = 0;
562bae44755Sopenharmony_ci    for (uint32_t i = 0; *(input + i) != '\0'; i++) {
563bae44755Sopenharmony_ci        if (*(input + i) == ',') {
564bae44755Sopenharmony_ci            count++;
565bae44755Sopenharmony_ci        }
566bae44755Sopenharmony_ci    }
567bae44755Sopenharmony_ci    count++;
568bae44755Sopenharmony_ci    // get private syscap string
569bae44755Sopenharmony_ci    char *priSysCapOut = (char *)malloc(SINGLE_SYSCAP_LEN * count);
570bae44755Sopenharmony_ci    if (priSysCapOut == NULL) {
571bae44755Sopenharmony_ci        PRINT_ERR("sscanf_s failed.\n");
572bae44755Sopenharmony_ci        return -1;
573bae44755Sopenharmony_ci    }
574bae44755Sopenharmony_ci    (void)memset_s(priSysCapOut, SINGLE_SYSCAP_LEN * count, 0, SINGLE_SYSCAP_LEN * count);
575bae44755Sopenharmony_ci    char *private = priSysCapOut;
576bae44755Sopenharmony_ci
577bae44755Sopenharmony_ci    char *tok = NULL;
578bae44755Sopenharmony_ci    char *temp = strtok_r(input, ",", &tok);
579bae44755Sopenharmony_ci    while (temp) {
580bae44755Sopenharmony_ci        int ret = strncpy_s(private, SINGLE_SYSCAP_LEN, temp, SINGLE_SYSCAP_LEN - 1);
581bae44755Sopenharmony_ci        if (ret != EOK) {
582bae44755Sopenharmony_ci            PRINT_ERR("strncpy_s failed.\n");
583bae44755Sopenharmony_ci            free(priSysCapOut);
584bae44755Sopenharmony_ci            return -1;
585bae44755Sopenharmony_ci        }
586bae44755Sopenharmony_ci        temp = strtok_r(NULL, ",", &tok);
587bae44755Sopenharmony_ci        private += SINGLE_SYSCAP_LEN;
588bae44755Sopenharmony_ci    }
589bae44755Sopenharmony_ci
590bae44755Sopenharmony_ci    *priSyscap = priSysCapOut;
591bae44755Sopenharmony_ci    *priSyscapLen = count;
592bae44755Sopenharmony_ci    return 0;
593bae44755Sopenharmony_ci}
594bae44755Sopenharmony_ci
595bae44755Sopenharmony_ciint32_t SeparateSyscapFromString(const char *inputString, uint32_t *osArray, uint32_t osArraySize,
596bae44755Sopenharmony_ci                                 char **priSyscap, uint32_t *priSyscapLen)
597bae44755Sopenharmony_ci{
598bae44755Sopenharmony_ci    if (osArraySize != PCID_OUT_BUFFER) {
599bae44755Sopenharmony_ci        return -1;
600bae44755Sopenharmony_ci    }
601bae44755Sopenharmony_ci
602bae44755Sopenharmony_ci    // copy origin string
603bae44755Sopenharmony_ci    char *input = CopyInputString(inputString);
604bae44755Sopenharmony_ci    if (input == NULL) {
605bae44755Sopenharmony_ci        return -1;
606bae44755Sopenharmony_ci    }
607bae44755Sopenharmony_ci
608bae44755Sopenharmony_ci    // get os syscap data
609bae44755Sopenharmony_ci    for (uint32_t i = 0; i < PCID_OUT_BUFFER; i++) {
610bae44755Sopenharmony_ci        if (sscanf_s(input, "%u,%s", &osArray[i], input, strlen(input)) == -1) {
611bae44755Sopenharmony_ci            PRINT_ERR("sscanf_s failed.\n");
612bae44755Sopenharmony_ci            free(input);
613bae44755Sopenharmony_ci            return -1;
614bae44755Sopenharmony_ci        }
615bae44755Sopenharmony_ci    }
616bae44755Sopenharmony_ci
617bae44755Sopenharmony_ci    // get private syscap data
618bae44755Sopenharmony_ci    if (GetPriSyscapData(input, priSyscap, priSyscapLen) != 0) {
619bae44755Sopenharmony_ci        free(input);
620bae44755Sopenharmony_ci        return -1;
621bae44755Sopenharmony_ci    }
622bae44755Sopenharmony_ci
623bae44755Sopenharmony_ci    free(input);
624bae44755Sopenharmony_ci    return 0;
625bae44755Sopenharmony_ci}
626bae44755Sopenharmony_ci
627bae44755Sopenharmony_ciint32_t GetSyscapByIndex(uint32_t index)
628bae44755Sopenharmony_ci{
629bae44755Sopenharmony_ci    const size_t allSyscapNum = sizeof(g_arraySyscap) / sizeof(SyscapWithNum);
630bae44755Sopenharmony_ci    for (uint32_t i = 0; i < allSyscapNum; i++) {
631bae44755Sopenharmony_ci        if (g_arraySyscap[i].num == index) {
632bae44755Sopenharmony_ci            return i;
633bae44755Sopenharmony_ci        }
634bae44755Sopenharmony_ci    }
635bae44755Sopenharmony_ci    return -1;
636bae44755Sopenharmony_ci}
637bae44755Sopenharmony_ci
638bae44755Sopenharmony_ciint32_t CompareOsSyscap(const uint32_t pcidOsArray[], const uint32_t rpcidOsAarry[])
639bae44755Sopenharmony_ci{
640bae44755Sopenharmony_ci    int32_t ossyscapFlag = 0;
641bae44755Sopenharmony_ci    const size_t allSyscapNum = sizeof(g_arraySyscap) / sizeof(SyscapWithNum);
642bae44755Sopenharmony_ci
643bae44755Sopenharmony_ci    for (uint32_t i = 2; i < PCID_OUT_BUFFER; i++) { // 2, header of pcid & rpcid
644bae44755Sopenharmony_ci        uint32_t blockBits = (pcidOsArray[i] ^ rpcidOsAarry[i]) & rpcidOsAarry[i];
645bae44755Sopenharmony_ci        if (!blockBits) {
646bae44755Sopenharmony_ci            continue;
647bae44755Sopenharmony_ci        }
648bae44755Sopenharmony_ci        for (uint8_t k = 0; k < INT_BIT; k++) {
649bae44755Sopenharmony_ci            if (!(blockBits & (1U << k))) {
650bae44755Sopenharmony_ci                continue;
651bae44755Sopenharmony_ci            }
652bae44755Sopenharmony_ci            // 2, header of pcid & rpcid
653bae44755Sopenharmony_ci            size_t pos = (size_t)((i - 2) * INT_BIT + k);
654bae44755Sopenharmony_ci            if (pos >= allSyscapNum) {
655bae44755Sopenharmony_ci                break;
656bae44755Sopenharmony_ci            }
657bae44755Sopenharmony_ci            printf("Missing: %s\n", g_arraySyscap[GetSyscapByIndex(pos)].str);
658bae44755Sopenharmony_ci            ossyscapFlag += 1;
659bae44755Sopenharmony_ci        }
660bae44755Sopenharmony_ci    }
661bae44755Sopenharmony_ci    return ossyscapFlag;
662bae44755Sopenharmony_ci}
663bae44755Sopenharmony_ci
664bae44755Sopenharmony_ciint32_t ComparePriSyscap(char *pcid, char *rpcid, uint32_t pcidLen, uint32_t rpcidLen)
665bae44755Sopenharmony_ci{
666bae44755Sopenharmony_ci    uint32_t i, j;
667bae44755Sopenharmony_ci    bool priSysFound = false;
668bae44755Sopenharmony_ci    int32_t prisyscapFlag = 0;
669bae44755Sopenharmony_ci
670bae44755Sopenharmony_ci    for (i = 0; i < rpcidLen; i++) {
671bae44755Sopenharmony_ci        for (j = 0; j < pcidLen; j++) {
672bae44755Sopenharmony_ci            if (strcmp(rpcid + SINGLE_SYSCAP_LEN * i,
673bae44755Sopenharmony_ci                       pcid + SINGLE_SYSCAP_LEN * j) == 0) {
674bae44755Sopenharmony_ci                priSysFound = true;
675bae44755Sopenharmony_ci                break;
676bae44755Sopenharmony_ci            }
677bae44755Sopenharmony_ci        }
678bae44755Sopenharmony_ci        if (priSysFound != true) {
679bae44755Sopenharmony_ci            printf("Missing: %s\n", rpcid + SINGLE_SYSCAP_LEN * i);
680bae44755Sopenharmony_ci            prisyscapFlag += 1;
681bae44755Sopenharmony_ci        }
682bae44755Sopenharmony_ci        priSysFound = false;
683bae44755Sopenharmony_ci    }
684bae44755Sopenharmony_ci
685bae44755Sopenharmony_ci    return prisyscapFlag;
686bae44755Sopenharmony_ci}
687bae44755Sopenharmony_ci
688bae44755Sopenharmony_cistatic int32_t CompareVersion(uint32_t *pcidOsArray, uint32_t *rpcidOsAarry)
689bae44755Sopenharmony_ci{
690bae44755Sopenharmony_ci    int32_t versionFlag = 0;
691bae44755Sopenharmony_ci    uint16_t pcidVersion = NtohsInter(((PCIDMain *)pcidOsArray)->apiVersion);
692bae44755Sopenharmony_ci    uint16_t rpcidVersion = NtohsInter(((RPCIDHead *)rpcidOsAarry)->apiVersion);
693bae44755Sopenharmony_ci    if (pcidVersion < rpcidVersion) {
694bae44755Sopenharmony_ci        printf("ERROR: Pcid version(%u) less than rpcid version(%u).\n", pcidVersion, rpcidVersion);
695bae44755Sopenharmony_ci        versionFlag = 1;
696bae44755Sopenharmony_ci    }
697bae44755Sopenharmony_ci    return versionFlag;
698bae44755Sopenharmony_ci}
699bae44755Sopenharmony_ci
700bae44755Sopenharmony_ciint32_t ComparePcidWithRpcidString(char *pcidFile, char *rpcidFile, uint32_t type)
701bae44755Sopenharmony_ci{
702bae44755Sopenharmony_ci    int32_t ret;
703bae44755Sopenharmony_ci    char *pcidContent = NULL;
704bae44755Sopenharmony_ci    char *rpcidContent = NULL;
705bae44755Sopenharmony_ci    char *pcidPriSyscap = NULL;
706bae44755Sopenharmony_ci    char *rpcidPriSyscap = NULL;
707bae44755Sopenharmony_ci    uint32_t pcidContentLen, rpcidContentLen, pcidPriSyscapLen, rpcidPriSyscapLen;
708bae44755Sopenharmony_ci    uint32_t pcidOsArray[PCID_OUT_BUFFER] = {0};
709bae44755Sopenharmony_ci    uint32_t rpcidOsAarry[PCID_OUT_BUFFER] = {0};
710bae44755Sopenharmony_ci
711bae44755Sopenharmony_ci    if (pcidFile == NULL || rpcidFile == NULL) {
712bae44755Sopenharmony_ci        PRINT_ERR("pcidFile or rpcidFile is null.\n");
713bae44755Sopenharmony_ci        return -1;
714bae44755Sopenharmony_ci    }
715bae44755Sopenharmony_ci
716bae44755Sopenharmony_ci    if (type == TYPE_FILE) {
717bae44755Sopenharmony_ci        if (GetFileContext(pcidFile, &pcidContent, &pcidContentLen)) {
718bae44755Sopenharmony_ci            PRINT_ERR("Get pcid file context failed, input file : %s\n", pcidFile);
719bae44755Sopenharmony_ci            return -1;
720bae44755Sopenharmony_ci        }
721bae44755Sopenharmony_ci        if (GetFileContext(rpcidFile, &rpcidContent, &rpcidContentLen)) {
722bae44755Sopenharmony_ci            PRINT_ERR("Get rpcid file context failed, input file : %s\n", rpcidFile);
723bae44755Sopenharmony_ci            free(pcidContent);
724bae44755Sopenharmony_ci            return -1;
725bae44755Sopenharmony_ci        }
726bae44755Sopenharmony_ci    } else if (type == TYPE_STRING) {
727bae44755Sopenharmony_ci        pcidContent = pcidFile;
728bae44755Sopenharmony_ci        rpcidContent = rpcidFile;
729bae44755Sopenharmony_ci    } else {
730bae44755Sopenharmony_ci        PRINT_ERR("Input file type error, type=%u\n", type);
731bae44755Sopenharmony_ci        return -1;
732bae44755Sopenharmony_ci    }
733bae44755Sopenharmony_ci
734bae44755Sopenharmony_ci    ret =  SeparateSyscapFromString(pcidContent, pcidOsArray, PCID_OUT_BUFFER,
735bae44755Sopenharmony_ci                                    &pcidPriSyscap, &pcidPriSyscapLen);
736bae44755Sopenharmony_ci    ret += SeparateSyscapFromString(rpcidContent, rpcidOsAarry, RPCID_OUT_BUFFER,
737bae44755Sopenharmony_ci                                    &rpcidPriSyscap, &rpcidPriSyscapLen);
738bae44755Sopenharmony_ci    if (ret != 0) {
739bae44755Sopenharmony_ci        PRINT_ERR("Separate syscap from string failed. ret = %d\n", ret);
740bae44755Sopenharmony_ci        FreePointerMemory(pcidContent, rpcidContent, pcidPriSyscap, rpcidPriSyscap);
741bae44755Sopenharmony_ci        return -1;
742bae44755Sopenharmony_ci    }
743bae44755Sopenharmony_ci
744bae44755Sopenharmony_ci    int32_t versionFlag = CompareVersion(pcidOsArray, rpcidOsAarry);
745bae44755Sopenharmony_ci    int32_t ossyscapFlag = CompareOsSyscap(pcidOsArray, rpcidOsAarry);
746bae44755Sopenharmony_ci    int32_t prisyscapFlag = ComparePriSyscap(pcidPriSyscap, rpcidPriSyscap, pcidPriSyscapLen, rpcidPriSyscapLen);
747bae44755Sopenharmony_ci    if (!versionFlag && !ossyscapFlag && !prisyscapFlag) {
748bae44755Sopenharmony_ci        printf("Succeed! The pcid meets the rpcid.\n");
749bae44755Sopenharmony_ci    } else {
750bae44755Sopenharmony_ci        printf("Fail! The pcid does not meet the rpcid\n");
751bae44755Sopenharmony_ci    }
752bae44755Sopenharmony_ci
753bae44755Sopenharmony_ci    return 0;
754bae44755Sopenharmony_ci}
755