/** * Copyright 2020 Huawei Technologies 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 "securec.h" /* * Find beginning of token (skip over leading delimiters).Note that * there is no token if this loop sets string to point to the terminal null. */ static char *SecFindBegin(char *strToken, const char *strDelimit) { char *token = strToken; while (*token != '\0') { const char *ctl = strDelimit; while (*ctl != '\0' && *ctl != *token) { ++ctl; } if (*ctl == '\0') { /* don't find any delimiter in string header, break the loop */ break; } ++token; } return token; } /* * Find rest of token */ static char *SecFindRest(char *strToken, const char *strDelimit) { /* Find the rest of the token. If it is not the end of the string, * put a null there. */ char *token = strToken; while (*token != '\0') { const char *ctl = strDelimit; while (*ctl != '\0' && *ctl != *token) { ++ctl; } if (*ctl != '\0') { /* find a delimiter */ *token++ = '\0'; /* set string termintor */ break; } ++token; } return token; } /* * Find the final position pointer */ static char *SecUpdateToken(char *strToken, const char *strDelimit, char **context) { /* point to updated position */ char *token = SecFindRest(strToken, strDelimit); /* record string position for next search in the context */ *context = token; /* Determine if a token has been found. */ if (token == strToken) { return NULL; } return strToken; } /* * * The strtok_s function parses a string into a sequence of strToken, * replace all characters in strToken string that match to strDelimit set with 0. * On the first call to strtok_s the string to be parsed should be specified in strToken. * In each subsequent call that should parse the same string, strToken should be NULL * * strToken String containing token or tokens. * strDelimit Set of delimiter characters. * context Used to store position information between calls * to strtok_s * * context is updated * * On the first call returns the address of the first non \0 character, otherwise NULL is returned. * In subsequent calls, the strtoken is set to NULL, and the context set is the same as the previous call, * return NULL if the *context string length is equal 0, otherwise return *context. */ char *strtok_s(char *strToken, const char *strDelimit, char **context) { char *orgToken = strToken; /* validate delimiter and string context */ if (context == NULL || strDelimit == NULL) { return NULL; } /* valid input string and string pointer from where to search */ if (orgToken == NULL && (*context) == NULL) { return NULL; } /* If string is null, continue searching from previous string position stored in context */ if (orgToken == NULL) { orgToken = *context; } orgToken = SecFindBegin(orgToken, strDelimit); return SecUpdateToken(orgToken, strDelimit, context); } #if SECUREC_IN_KERNEL EXPORT_SYMBOL(strtok_s); #endif