mirror of
https://github.com/alliedmodders/amxmodx.git
synced 2025-01-01 01:35:37 +03:00
147 lines
4.5 KiB
C
147 lines
4.5 KiB
C
|
/*
|
||
|
Copyright (C) 2014-2016 Quinten Lansu
|
||
|
|
||
|
Permission is hereby granted, free of charge, to any person
|
||
|
obtaining a copy of this software and associated documentation
|
||
|
files (the "Software"), to deal in the Software without
|
||
|
restriction, including without limitation the rights to use,
|
||
|
copy, modify, merge, publish, distribute, sublicense, and/or
|
||
|
sell copies of the Software, and to permit persons to whom the
|
||
|
Software is furnished to do so, subject to the following
|
||
|
conditions:
|
||
|
|
||
|
The above copyright notice and this permission notice shall be
|
||
|
included in all copies or substantial portions of the Software.
|
||
|
|
||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
||
|
EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES
|
||
|
OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
||
|
NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT
|
||
|
HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
|
||
|
WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
|
||
|
FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
|
||
|
OTHER DEALINGS IN THE SOFTWARE.
|
||
|
*/
|
||
|
|
||
|
#ifndef _UTF8REWIND_INTERNAL_BASE_H_
|
||
|
#define _UTF8REWIND_INTERNAL_BASE_H_
|
||
|
|
||
|
/*!
|
||
|
\file
|
||
|
\brief Base header for internal interface.
|
||
|
|
||
|
\cond INTERNAL
|
||
|
*/
|
||
|
|
||
|
#include "utf8rewind.h"
|
||
|
|
||
|
#if defined(__GNUC__) && !defined(COMPILER_ICC)
|
||
|
#define UTF8_UNUSED(_parameter) _parameter __attribute__ ((unused))
|
||
|
#else
|
||
|
#define UTF8_UNUSED(_parameter) _parameter
|
||
|
#endif
|
||
|
|
||
|
#define UTF8_SET_ERROR(_error) \
|
||
|
if (errors != 0) { *errors = UTF8_ERR_ ## _error; }
|
||
|
|
||
|
/* Validates input before transforming */
|
||
|
/* Check for parameter overlap using the separating axis theorem */
|
||
|
|
||
|
#define UTF8_VALIDATE_PARAMETERS_CHAR(_inputType, _result) \
|
||
|
if (input == 0) { \
|
||
|
UTF8_SET_ERROR(INVALID_DATA); \
|
||
|
return _result; \
|
||
|
} \
|
||
|
else if (inputSize < sizeof(_inputType)) { \
|
||
|
if (target != 0) { \
|
||
|
if (targetSize < 3) { \
|
||
|
UTF8_SET_ERROR(NOT_ENOUGH_SPACE); \
|
||
|
return _result; \
|
||
|
} \
|
||
|
memcpy(target, REPLACEMENT_CHARACTER_STRING, REPLACEMENT_CHARACTER_STRING_LENGTH); \
|
||
|
} \
|
||
|
UTF8_SET_ERROR(INVALID_DATA); \
|
||
|
return _result + REPLACEMENT_CHARACTER_STRING_LENGTH; \
|
||
|
} \
|
||
|
if (target != 0 && targetSize == 0) { \
|
||
|
UTF8_SET_ERROR(NOT_ENOUGH_SPACE); \
|
||
|
return _result; \
|
||
|
} \
|
||
|
if ((char*)input == target) { \
|
||
|
UTF8_SET_ERROR(OVERLAPPING_PARAMETERS); \
|
||
|
return _result; \
|
||
|
} \
|
||
|
{ \
|
||
|
char* input_center = (char*)input + (inputSize / 2); \
|
||
|
char* target_center = target + (targetSize / 2); \
|
||
|
size_t delta = (size_t)((input_center > target_center) ? (input_center - target_center) : (target_center - input_center)); \
|
||
|
if (delta < (inputSize + targetSize) / 2) { \
|
||
|
UTF8_SET_ERROR(OVERLAPPING_PARAMETERS); \
|
||
|
return _result; \
|
||
|
} \
|
||
|
}
|
||
|
|
||
|
#define UTF8_VALIDATE_PARAMETERS_CHAR_NOCR(_inputType, _result) \
|
||
|
if (input == 0) { \
|
||
|
UTF8_SET_ERROR(INVALID_DATA); \
|
||
|
return _result; \
|
||
|
} \
|
||
|
else if (inputSize < sizeof(_inputType)) { \
|
||
|
UTF8_SET_ERROR(INVALID_DATA); \
|
||
|
return _result; \
|
||
|
} \
|
||
|
if (target != 0 && targetSize == 0) { \
|
||
|
UTF8_SET_ERROR(NOT_ENOUGH_SPACE); \
|
||
|
return _result; \
|
||
|
} \
|
||
|
if ((char*)input == target) { \
|
||
|
UTF8_SET_ERROR(OVERLAPPING_PARAMETERS); \
|
||
|
return _result; \
|
||
|
} \
|
||
|
{ \
|
||
|
char* input_center = (char*)input + (inputSize / 2); \
|
||
|
char* target_center = target + (targetSize / 2); \
|
||
|
size_t delta = (size_t)((input_center > target_center) ? (input_center - target_center) : (target_center - input_center)); \
|
||
|
if (delta < (inputSize + targetSize) / 2) { \
|
||
|
UTF8_SET_ERROR(OVERLAPPING_PARAMETERS); \
|
||
|
return _result; \
|
||
|
} \
|
||
|
}
|
||
|
|
||
|
#define UTF8_VALIDATE_PARAMETERS(_inputType, _outputType, _result) \
|
||
|
if (input == 0) { \
|
||
|
UTF8_SET_ERROR(INVALID_DATA); \
|
||
|
return _result; \
|
||
|
} \
|
||
|
else if (inputSize < sizeof(_inputType)) { \
|
||
|
if (target != 0) { \
|
||
|
if (targetSize < sizeof(_outputType)) { \
|
||
|
UTF8_SET_ERROR(NOT_ENOUGH_SPACE); \
|
||
|
return _result; \
|
||
|
} \
|
||
|
*target = REPLACEMENT_CHARACTER; \
|
||
|
} \
|
||
|
UTF8_SET_ERROR(INVALID_DATA); \
|
||
|
return _result + sizeof(_outputType); \
|
||
|
} \
|
||
|
if (target != 0 && targetSize < sizeof(_outputType)) { \
|
||
|
UTF8_SET_ERROR(NOT_ENOUGH_SPACE); \
|
||
|
return _result; \
|
||
|
} \
|
||
|
if ((char*)input == (char*)target) { \
|
||
|
UTF8_SET_ERROR(OVERLAPPING_PARAMETERS); \
|
||
|
return _result; \
|
||
|
} \
|
||
|
{ \
|
||
|
char* input_center = (char*)input + (inputSize / 2); \
|
||
|
char* target_center = (char*)target + (targetSize / 2); \
|
||
|
size_t delta = (size_t)((input_center > target_center) ? (input_center - target_center) : (target_center - input_center)); \
|
||
|
if (delta < (inputSize + targetSize) / 2) { \
|
||
|
UTF8_SET_ERROR(OVERLAPPING_PARAMETERS); \
|
||
|
return _result; \
|
||
|
} \
|
||
|
}
|
||
|
|
||
|
/*! \endcond */
|
||
|
|
||
|
#endif /* _UTF8REWIND_INTERNAL_BASE_H_ */
|