osString.c 8.8 KB
Newer Older
S
Shengliang Guan 已提交
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15
/*
 * Copyright (c) 2019 TAOS Data, Inc. <jhtao@taosdata.com>
 *
 * This program is free software: you can use, redistribute, and/or modify
 * it under the terms of the GNU Affero General Public License, version 3
 * or later ("AGPL"), as published by the Free Software Foundation.
 *
 * This program is distributed in the hope that it will be useful, but WITHOUT
 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
 * FITNESS FOR A PARTICULAR PURPOSE.
 *
 * You should have received a copy of the GNU Affero General Public License
 * along with this program. If not, see <http://www.gnu.org/licenses/>.
 */

wafwerar's avatar
wafwerar 已提交
16
#define ALLOW_FORBID_FUNC
S
Shengliang Guan 已提交
17 18 19
#define _DEFAULT_SOURCE
#include "os.h"

wafwerar's avatar
wafwerar 已提交
20 21 22
#ifndef DISALLOW_NCHAR_WITHOUT_ICONV
#include "iconv.h"
#endif
S
Shengliang Guan 已提交
23

wafwerar's avatar
wafwerar 已提交
24 25 26
extern int wcwidth(wchar_t c);
extern int wcswidth(const wchar_t *s, size_t n);

wafwerar's avatar
wafwerar 已提交
27 28 29 30 31
#ifdef WINDOWS
char *strsep(char **stringp, const char *delim) {
  char *      s;
  const char *spanp;
  int32_t     c, sc;
dengyihao's avatar
dengyihao 已提交
32 33
  char *      tok;
  if ((s = *stringp) == NULL) return (NULL);
wafwerar's avatar
wafwerar 已提交
34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52
  for (tok = s;;) {
    c = *s++;
    spanp = delim;
    do {
      if ((sc = *spanp++) == c) {
        if (c == 0)
          s = NULL;
        else
          s[-1] = 0;
        *stringp = s;
        return (tok);
      }
    } while (sc != 0);
  }
  /* NOTREACHED */
}
/* Duplicate a string, up to at most size characters */
char *strndup(const char *s, size_t size) {
  size_t l;
dengyihao's avatar
dengyihao 已提交
53
  char * s2;
wafwerar's avatar
wafwerar 已提交
54
  l = strlen(s);
dengyihao's avatar
dengyihao 已提交
55 56
  if (l > size) l = size;
  s2 = malloc(l + 1);
wafwerar's avatar
wafwerar 已提交
57 58 59 60 61 62 63 64
  if (s2) {
    strncpy(s2, s, l);
    s2[l] = '\0';
  }
  return s2;
}
/* Copy no more than N characters of SRC to DEST, returning the address of
   the terminating '\0' in DEST, if any, or else DEST + N.  */
dengyihao's avatar
dengyihao 已提交
65 66 67
char *stpncpy(char *dest, const char *src, size_t n) {
  size_t size = strnlen(src, n);
  memcpy(dest, src, size);
wafwerar's avatar
wafwerar 已提交
68
  dest += size;
dengyihao's avatar
dengyihao 已提交
69 70
  if (size == n) return dest;
  return memset(dest, '\0', n - size);
wafwerar's avatar
wafwerar 已提交
71 72 73
}
#endif

74
int64_t taosStr2int64(const char *str) {
S
Shengliang Guan 已提交
75 76 77 78
  char *endptr = NULL;
  return strtoll(str, &endptr, 10);
}

wafwerar's avatar
wafwerar 已提交
79 80 81 82 83 84 85 86 87 88 89 90 91 92 93
int32_t tasoUcs4Compare(TdUcs4 *f1_ucs4, TdUcs4 *f2_ucs4, int32_t bytes) {
  for (int32_t i = 0; i < bytes; i += sizeof(TdUcs4)) {
    int32_t f1 = *(int32_t *)((char *)f1_ucs4 + i);
    int32_t f2 = *(int32_t *)((char *)f2_ucs4 + i);

    if ((f1 == 0 && f2 != 0) || (f1 != 0 && f2 == 0)) {
      return f1 - f2;
    } else if (f1 == 0 && f2 == 0) {
      return 0;
    }

    if (f1 != f2) {
      return f1 - f2;
    }
  }
S
Shengliang Guan 已提交
94

wafwerar's avatar
wafwerar 已提交
95 96 97 98
  return 0;

#if 0
  int32_t ucs4_max_len = bytes + 4;
wafwerar's avatar
wafwerar 已提交
99 100
  char *f1_mbs = taosMemoryCalloc(bytes, 1);
  char *f2_mbs = taosMemoryCalloc(bytes, 1);
wafwerar's avatar
wafwerar 已提交
101 102 103 104 105 106 107
  if (taosUcs4ToMbs(f1_ucs4, ucs4_max_len, f1_mbs) < 0) {
    return -1;
  }
  if (taosUcs4ToMbs(f2_ucs4, ucs4_max_len, f2_mbs) < 0) {
    return -1;
  }
  int32_t ret = strcmp(f1_mbs, f2_mbs);
wafwerar's avatar
wafwerar 已提交
108 109
  taosMemoryFree(f1_mbs);
  taosMemoryFree(f2_mbs);
wafwerar's avatar
wafwerar 已提交
110 111 112 113
  return ret;
#endif
}

dengyihao's avatar
dengyihao 已提交
114 115 116
TdUcs4 *tasoUcs4Copy(TdUcs4 *target_ucs4, TdUcs4 *source_ucs4, int32_t len_ucs4) {
  assert(taosMemorySize(target_ucs4) >= len_ucs4 * sizeof(TdUcs4));
  return memcpy(target_ucs4, source_ucs4, len_ucs4 * sizeof(TdUcs4));
wafwerar's avatar
wafwerar 已提交
117 118
}

wafwerar's avatar
wafwerar 已提交
119 120
int32_t taosUcs4ToMbs(TdUcs4 *ucs4, int32_t ucs4_max_len, char *mbs) {
#ifdef DISALLOW_NCHAR_WITHOUT_ICONV
121
  printf("Nchar cannot be read and written without iconv, please install iconv library and recompile TDengine.\n");
wafwerar's avatar
wafwerar 已提交
122 123
  return -1;
#else
S
Shengliang Guan 已提交
124 125 126 127 128 129 130 131 132 133
  iconv_t cd = iconv_open(tsCharset, DEFAULT_UNICODE_ENCODEC);
  size_t  ucs4_input_len = ucs4_max_len;
  size_t  outLen = ucs4_max_len;
  if (iconv(cd, (char **)&ucs4, &ucs4_input_len, &mbs, &outLen) == -1) {
    iconv_close(cd);
    return -1;
  }

  iconv_close(cd);
  return (int32_t)(ucs4_max_len - outLen);
wafwerar's avatar
wafwerar 已提交
134
#endif
S
Shengliang Guan 已提交
135 136
}

dengyihao's avatar
dengyihao 已提交
137
bool taosMbsToUcs4(const char *mbs, size_t mbsLength, TdUcs4 *ucs4, int32_t ucs4_max_len, int32_t *len) {
wafwerar's avatar
wafwerar 已提交
138
#ifdef DISALLOW_NCHAR_WITHOUT_ICONV
139
  printf("Nchar cannot be read and written without iconv, please install iconv library and recompile TDengine.\n");
wafwerar's avatar
wafwerar 已提交
140 141
  return -1;
#else
S
Shengliang Guan 已提交
142 143 144 145
  memset(ucs4, 0, ucs4_max_len);
  iconv_t cd = iconv_open(DEFAULT_UNICODE_ENCODEC, tsCharset);
  size_t  ucs4_input_len = mbsLength;
  size_t  outLeft = ucs4_max_len;
dengyihao's avatar
dengyihao 已提交
146
  if (iconv(cd, (char **)&mbs, &ucs4_input_len, (char **)&ucs4, &outLeft) == -1) {
S
Shengliang Guan 已提交
147 148 149 150 151 152 153 154 155 156 157 158 159
    iconv_close(cd);
    return false;
  }

  iconv_close(cd);
  if (len != NULL) {
    *len = (int32_t)(ucs4_max_len - outLeft);
    if (*len < 0) {
      return false;
    }
  }

  return true;
wafwerar's avatar
wafwerar 已提交
160
#endif
S
Shengliang Guan 已提交
161 162 163
}

bool taosValidateEncodec(const char *encodec) {
wafwerar's avatar
wafwerar 已提交
164
#ifdef DISALLOW_NCHAR_WITHOUT_ICONV
165 166
  printf("Nchar cannot be read and written without iconv, please install iconv library and recompile TDengine.\n");
  return true;
wafwerar's avatar
wafwerar 已提交
167
#else
S
Shengliang Guan 已提交
168 169 170 171 172 173 174 175 176 177
  iconv_t cd = iconv_open(encodec, DEFAULT_UNICODE_ENCODEC);
  if (cd == (iconv_t)(-1)) {
    return false;
  }

  iconv_close(cd);
  return true;
#endif
}

wafwerar's avatar
wafwerar 已提交
178 179
int32_t taosUcs4len(TdUcs4 *ucs4) {
  TdUcs4 *wstr = (TdUcs4 *)ucs4;
S
Shengliang Guan 已提交
180 181 182 183 184 185 186 187 188 189 190 191 192 193 194
  if (NULL == wstr) {
    return 0;
  }

  int32_t n = 0;
  while (1) {
    if (0 == *wstr++) {
      break;
    }
    n++;
  }

  return n;
}

dengyihao's avatar
dengyihao 已提交
195
// dst buffer size should be at least 2*len + 1
wafwerar's avatar
wafwerar 已提交
196
int32_t taosHexEncode(const unsigned char *src, char *dst, int32_t len) {
197 198 199 200 201
  if (!dst) {
    return -1;
  }

  for (int32_t i = 0; i < len; ++i) {
wmmhello's avatar
wmmhello 已提交
202
    sprintf(dst + i * 2, "%02x", src[i]);
203 204 205 206 207 208 209 210 211 212
  }

  return 0;
}

int32_t taosHexDecode(const char *src, char *dst, int32_t len) {
  if (!dst) {
    return -1;
  }

wmmhello's avatar
wmmhello 已提交
213
  uint8_t hn, ln, out;
dengyihao's avatar
dengyihao 已提交
214
  for (int i = 0, j = 0; i < len * 2; i += 2, ++j) {
wmmhello's avatar
wmmhello 已提交
215 216
    hn = src[i] > '9' ? src[i] - 'a' + 10 : src[i] - '0';
    ln = src[i + 1] > '9' ? src[i + 1] - 'a' + 10 : src[i + 1] - '0';
217 218 219 220 221 222 223 224

    out = (hn << 4) | ln;
    memcpy(dst + j, &out, 1);
  }

  return 0;
}

wafwerar's avatar
wafwerar 已提交
225 226 227 228 229 230 231 232 233 234 235
int32_t taosWcharWidth(TdWchar wchar) { return wcwidth(wchar); }

int32_t taosWcharsWidth(TdWchar *pWchar, int32_t size) { return wcswidth(pWchar, size); }

int32_t taosMbToWchar(TdWchar *pWchar, const char *pStr, int32_t size) { return mbtowc(pWchar, pStr, size); }

int32_t taosMbsToWchars(TdWchar *pWchars, const char *pStrs, int32_t size) { return mbstowcs(pWchars, pStrs, size); }

int32_t taosWcharToMb(char *pStr, TdWchar wchar) { return wctomb(pStr, wchar); }

int32_t taosWcharsToMbs(char *pStrs, TdWchar *pWchars, int32_t size) { return wcstombs(pStrs, pWchars, size); }
wafwerar's avatar
wafwerar 已提交
236 237

char *taosStrCaseStr(const char *str, const char *pattern) {
dengyihao's avatar
dengyihao 已提交
238 239 240 241 242 243 244 245 246 247 248 249 250
  size_t i;

  if (!*pattern) return (char *)str;

  for (; *str; str++) {
    if (toupper(*str) == toupper(*pattern)) {
      for (i = 1;; i++) {
        if (!pattern[i]) return (char *)str;
        if (toupper(str[i]) != toupper(pattern[i])) break;
      }
    }
  }
  return NULL;
wafwerar's avatar
wafwerar 已提交
251 252
}

dengyihao's avatar
dengyihao 已提交
253
int64_t taosStr2Int64(const char *str, char **pEnd, int32_t radix) {
wafwerar's avatar
wafwerar 已提交
254
  int64_t tmp = strtoll(str, pEnd, radix);
wafwerar's avatar
wafwerar 已提交
255
#ifdef TD_CHECK_STR_TO_INT_ERROR
wafwerar's avatar
wafwerar 已提交
256 257
  assert(errno != ERANGE);
  assert(errno != EINVAL);
wafwerar's avatar
wafwerar 已提交
258
#endif
wafwerar's avatar
wafwerar 已提交
259 260 261
  return tmp;
}

dengyihao's avatar
dengyihao 已提交
262
uint64_t taosStr2UInt64(const char *str, char **pEnd, int32_t radix) {
wafwerar's avatar
wafwerar 已提交
263
  uint64_t tmp = strtoull(str, pEnd, radix);
wafwerar's avatar
wafwerar 已提交
264
#ifdef TD_CHECK_STR_TO_INT_ERROR
wafwerar's avatar
wafwerar 已提交
265 266
  assert(errno != ERANGE);
  assert(errno != EINVAL);
wafwerar's avatar
wafwerar 已提交
267
#endif
wafwerar's avatar
wafwerar 已提交
268 269 270
  return tmp;
}

dengyihao's avatar
dengyihao 已提交
271
int32_t taosStr2Int32(const char *str, char **pEnd, int32_t radix) {
wafwerar's avatar
wafwerar 已提交
272
  int32_t tmp = strtol(str, pEnd, radix);
wafwerar's avatar
wafwerar 已提交
273
#ifdef TD_CHECK_STR_TO_INT_ERROR
wafwerar's avatar
wafwerar 已提交
274 275
  assert(errno != ERANGE);
  assert(errno != EINVAL);
wafwerar's avatar
wafwerar 已提交
276
#endif
wafwerar's avatar
wafwerar 已提交
277 278 279
  return tmp;
}

dengyihao's avatar
dengyihao 已提交
280
uint32_t taosStr2UInt32(const char *str, char **pEnd, int32_t radix) {
wafwerar's avatar
wafwerar 已提交
281
  uint32_t tmp = strtol(str, pEnd, radix);
wafwerar's avatar
wafwerar 已提交
282
#ifdef TD_CHECK_STR_TO_INT_ERROR
wafwerar's avatar
wafwerar 已提交
283 284
  assert(errno != ERANGE);
  assert(errno != EINVAL);
wafwerar's avatar
wafwerar 已提交
285
#endif
wafwerar's avatar
wafwerar 已提交
286 287 288
  return tmp;
}

dengyihao's avatar
dengyihao 已提交
289
int16_t taosStr2Int16(const char *str, char **pEnd, int32_t radix) {
wafwerar's avatar
wafwerar 已提交
290
  int32_t tmp = strtol(str, pEnd, radix);
wafwerar's avatar
wafwerar 已提交
291
#ifdef TD_CHECK_STR_TO_INT_ERROR
wafwerar's avatar
wafwerar 已提交
292 293 294 295
  assert(errno != ERANGE);
  assert(errno != EINVAL);
  assert(tmp >= SHRT_MIN);
  assert(tmp <= SHRT_MAX);
wafwerar's avatar
wafwerar 已提交
296
#endif
wafwerar's avatar
wafwerar 已提交
297 298 299
  return (int16_t)tmp;
}

dengyihao's avatar
dengyihao 已提交
300
uint16_t taosStr2UInt16(const char *str, char **pEnd, int32_t radix) {
wafwerar's avatar
wafwerar 已提交
301
  uint32_t tmp = strtoul(str, pEnd, radix);
wafwerar's avatar
wafwerar 已提交
302
#ifdef TD_CHECK_STR_TO_INT_ERROR
wafwerar's avatar
wafwerar 已提交
303 304 305
  assert(errno != ERANGE);
  assert(errno != EINVAL);
  assert(tmp <= USHRT_MAX);
wafwerar's avatar
wafwerar 已提交
306
#endif
wafwerar's avatar
wafwerar 已提交
307 308 309
  return (uint16_t)tmp;
}

dengyihao's avatar
dengyihao 已提交
310
int8_t taosStr2Int8(const char *str, char **pEnd, int32_t radix) {
wafwerar's avatar
wafwerar 已提交
311
  int32_t tmp = strtol(str, pEnd, radix);
wafwerar's avatar
wafwerar 已提交
312
#ifdef TD_CHECK_STR_TO_INT_ERROR
wafwerar's avatar
wafwerar 已提交
313 314 315 316
  assert(errno != ERANGE);
  assert(errno != EINVAL);
  assert(tmp >= SCHAR_MIN);
  assert(tmp <= SCHAR_MAX);
wafwerar's avatar
wafwerar 已提交
317
#endif
wafwerar's avatar
wafwerar 已提交
318 319 320
  return tmp;
}

dengyihao's avatar
dengyihao 已提交
321
uint8_t taosStr2UInt8(const char *str, char **pEnd, int32_t radix) {
wafwerar's avatar
wafwerar 已提交
322
  uint32_t tmp = strtoul(str, pEnd, radix);
wafwerar's avatar
wafwerar 已提交
323
#ifdef TD_CHECK_STR_TO_INT_ERROR
wafwerar's avatar
wafwerar 已提交
324 325 326
  assert(errno != ERANGE);
  assert(errno != EINVAL);
  assert(tmp <= UCHAR_MAX);
wafwerar's avatar
wafwerar 已提交
327
#endif
wafwerar's avatar
wafwerar 已提交
328 329 330
  return tmp;
}

dengyihao's avatar
dengyihao 已提交
331
double taosStr2Double(const char *str, char **pEnd) {
wafwerar's avatar
wafwerar 已提交
332
  double tmp = strtod(str, pEnd);
wafwerar's avatar
wafwerar 已提交
333
#ifdef TD_CHECK_STR_TO_INT_ERROR
wafwerar's avatar
wafwerar 已提交
334 335 336
  assert(errno != ERANGE);
  assert(errno != EINVAL);
  assert(tmp != HUGE_VAL);
wafwerar's avatar
wafwerar 已提交
337
#endif
wafwerar's avatar
wafwerar 已提交
338 339 340
  return tmp;
}

dengyihao's avatar
dengyihao 已提交
341
float taosStr2Float(const char *str, char **pEnd) {
wafwerar's avatar
wafwerar 已提交
342
  float tmp = strtof(str, pEnd);
wafwerar's avatar
wafwerar 已提交
343
#ifdef TD_CHECK_STR_TO_INT_ERROR
wafwerar's avatar
wafwerar 已提交
344 345 346 347
  assert(errno != ERANGE);
  assert(errno != EINVAL);
  assert(tmp != HUGE_VALF);
  assert(tmp != NAN);
wafwerar's avatar
wafwerar 已提交
348
#endif
wafwerar's avatar
wafwerar 已提交
349
  return tmp;
dengyihao's avatar
dengyihao 已提交
350
}