|
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120 |
- /**
- * 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 "securecutil.h"
-
- #ifdef SECUREC_NOT_CALL_LIBC_CORE_API
- /*
- * Implementing memory data movement
- */
- static void SecUtilMemmove(void *dst, const void *src, size_t count)
- {
- unsigned char *pDest = (unsigned char *)dst;
- const unsigned char *pSrc = (const unsigned char *)src;
- size_t maxCount = count;
-
- if (dst <= src || pDest >= (pSrc + maxCount)) {
- /*
- * Non-Overlapping Buffers
- * copy from lower addresses to higher addresses
- */
- while (maxCount--) {
- *pDest = *pSrc;
- ++pDest;
- ++pSrc;
- }
- } else {
- /*
- * Overlapping Buffers
- * copy from higher addresses to lower addresses
- */
- pDest = pDest + maxCount - 1;
- pSrc = pSrc + maxCount - 1;
-
- while (maxCount--) {
- *pDest = *pSrc;
-
- --pDest;
- --pSrc;
- }
- }
- }
- #endif
-
- /*
- * <FUNCTION DESCRIPTION>
- * The memmove_s function copies count bytes of characters from src to dest.
- * This function can be assigned correctly when memory overlaps.
- * <INPUT PARAMETERS>
- * dest Destination object.
- * destMax Size of the destination buffer.
- * src Source object.
- * count Number of characters to copy.
- *
- * <OUTPUT PARAMETERS>
- * dest buffer is uptdated.
- *
- * <RETURN VALUE>
- * EOK Success
- * EINVAL dest is NULL and destMax != 0 and destMax <= SECUREC_MEM_MAX_LEN
- * EINVAL_AND_RESET dest != NULL and src is NULLL and destMax != 0 and destMax <= SECUREC_MEM_MAX_LEN
- * ERANGE destMax > SECUREC_MEM_MAX_LEN or destMax is 0
- * ERANGE_AND_RESET count > destMax and dest != NULL and src != NULL and destMax != 0
- * and destMax <= SECUREC_MEM_MAX_LEN
- *
- * If an error occured, dest will be filled with 0 when dest and destMax valid.
- * If some regions of the source area and the destination overlap, memmove_s
- * ensures that the original source bytes in the overlapping region are copied
- * before being overwritten.
- */
- errno_t memmove_s(void *dest, size_t destMax, const void *src, size_t count)
- {
- if (destMax == 0 || destMax > SECUREC_MEM_MAX_LEN) {
- SECUREC_ERROR_INVALID_RANGE("memmove_s");
- return ERANGE;
- }
- if (dest == NULL || src == NULL) {
- SECUREC_ERROR_INVALID_PARAMTER("memmove_s");
- if (dest != NULL) {
- (void)memset(dest, 0, destMax);
- return EINVAL_AND_RESET;
- }
- return EINVAL;
- }
- if (count > destMax) {
- (void)memset(dest, 0, destMax);
- SECUREC_ERROR_INVALID_RANGE("memmove_s");
- return ERANGE_AND_RESET;
- }
- if (dest == src) {
- return EOK;
- }
-
- if (count > 0) {
- #ifdef SECUREC_NOT_CALL_LIBC_CORE_API
- SecUtilMemmove(dest, src, count);
- #else
- /* use underlying memmove for performance consideration */
- (void)memmove(dest, src, count);
- #endif
- }
- return EOK;
- }
-
- #if SECUREC_IN_KERNEL
- EXPORT_SYMBOL(memmove_s);
- #endif
-
|