Home | History | Annotate | Download | only in genrb
      1 //  2016 and later: Unicode, Inc. and others.
      2 // License & terms of use: http://www.unicode.org/copyright.html
      3 /*
      4 *******************************************************************************
      5 *
      6 *   Copyright (C) 1998-2012, International Business Machines
      7 *   Corporation and others.  All Rights Reserved.
      8 *
      9 *******************************************************************************
     10 *
     11 * File ustr.c
     12 *
     13 * Modification History:
     14 *
     15 *   Date        Name        Description
     16 *   05/28/99    stephen     Creation.
     17 *******************************************************************************
     18 */
     19 
     20 #include "ustr.h"
     21 #include "cmemory.h"
     22 #include "cstring.h"
     23 #include "unicode/ustring.h"
     24 #include "unicode/putil.h"
     25 #include "unicode/utf16.h"
     26 
     27 /* Protos */
     28 static void ustr_resize(struct UString *s, int32_t len, UErrorCode *status);
     29 
     30 /* Macros */
     31 #define ALLOCATION(minSize) (minSize < 0x80 ? 0x80 : (2 * minSize + 0x80) & ~(0x80 - 1))
     32 
     33 U_CFUNC void
     34 ustr_init(struct UString *s)
     35 {
     36     s->fChars = 0;
     37     s->fLength = s->fCapacity = 0;
     38 }
     39 
     40 U_CFUNC void
     41 ustr_initChars(struct UString *s, const char* source, int32_t length, UErrorCode *status)
     42 {
     43     int i = 0;
     44     if (U_FAILURE(*status)) return;
     45     s->fChars = 0;
     46     s->fLength = s->fCapacity = 0;
     47     if (length == -1) {
     48         length = (int32_t)uprv_strlen(source);
     49     }
     50     if(s->fCapacity < length) {
     51       ustr_resize(s, ALLOCATION(length), status);
     52       if(U_FAILURE(*status)) return;
     53     }
     54     for (; i < length; i++)
     55     {
     56       UChar charToAppend;
     57       u_charsToUChars(source+i, &charToAppend, 1);
     58       ustr_ucat(s, charToAppend, status);
     59       /*
     60 #if U_CHARSET_FAMILY==U_ASCII_FAMILY
     61         ustr_ucat(s, (UChar)(uint8_t)(source[i]), status);
     62 #elif U_CHARSET_FAMILY==U_EBCDIC_FAMILY
     63         ustr_ucat(s, (UChar)asciiFromEbcdic[(uint8_t)(*cs++)], status);
     64 #else
     65 #   error U_CHARSET_FAMILY is not valid
     66 #endif
     67       */
     68     }
     69 }
     70 
     71 U_CFUNC void
     72 ustr_deinit(struct UString *s)
     73 {
     74     if (s) {
     75         uprv_free(s->fChars);
     76         s->fChars = 0;
     77         s->fLength = s->fCapacity = 0;
     78     }
     79 }
     80 
     81 U_CFUNC void
     82 ustr_cpy(struct UString *dst,
     83      const struct UString *src,
     84      UErrorCode *status)
     85 {
     86     if(U_FAILURE(*status) || dst == src)
     87         return;
     88 
     89     if(dst->fCapacity < src->fLength) {
     90         ustr_resize(dst, ALLOCATION(src->fLength), status);
     91         if(U_FAILURE(*status))
     92             return;
     93     }
     94     if(src->fChars == NULL || dst->fChars == NULL){
     95         return;
     96     }
     97     u_memcpy(dst->fChars, src->fChars, src->fLength);
     98     dst->fLength = src->fLength;
     99     dst->fChars[dst->fLength] = 0x0000;
    100 }
    101 
    102 U_CFUNC void
    103 ustr_setlen(struct UString *s,
    104         int32_t len,
    105         UErrorCode *status)
    106 {
    107     if(U_FAILURE(*status))
    108         return;
    109 
    110     if(s->fCapacity < (len + 1)) {
    111         ustr_resize(s, ALLOCATION(len), status);
    112         if(U_FAILURE(*status))
    113             return;
    114     }
    115 
    116     s->fLength = len;
    117     s->fChars[len] = 0x0000;
    118 }
    119 
    120 U_CFUNC void
    121 ustr_cat(struct UString *dst,
    122      const struct UString *src,
    123      UErrorCode *status)
    124 {
    125     ustr_ncat(dst, src, src->fLength, status);
    126 }
    127 
    128 U_CFUNC void
    129 ustr_ncat(struct UString *dst,
    130       const struct UString *src,
    131       int32_t n,
    132       UErrorCode *status)
    133 {
    134     if(U_FAILURE(*status) || dst == src)
    135         return;
    136 
    137     if(dst->fCapacity < (dst->fLength + n)) {
    138         ustr_resize(dst, ALLOCATION(dst->fLength + n), status);
    139         if(U_FAILURE(*status))
    140             return;
    141     }
    142 
    143     uprv_memcpy(dst->fChars + dst->fLength, src->fChars,
    144                 sizeof(UChar) * n);
    145     dst->fLength += src->fLength;
    146     dst->fChars[dst->fLength] = 0x0000;
    147 }
    148 
    149 U_CFUNC void
    150 ustr_ucat(struct UString *dst,
    151       UChar c,
    152       UErrorCode *status)
    153 {
    154     if(U_FAILURE(*status))
    155         return;
    156 
    157     if(dst->fCapacity < (dst->fLength + 1)) {
    158         ustr_resize(dst, ALLOCATION(dst->fLength + 1), status);
    159         if(U_FAILURE(*status))
    160             return;
    161     }
    162 
    163     uprv_memcpy(dst->fChars + dst->fLength, &c,
    164         sizeof(UChar) * 1);
    165     dst->fLength += 1;
    166     dst->fChars[dst->fLength] = 0x0000;
    167 }
    168 U_CFUNC void
    169 ustr_u32cat(struct UString *dst, UChar32 c, UErrorCode *status){
    170     if(c > 0x10FFFF){
    171         *status = U_ILLEGAL_CHAR_FOUND;
    172         return;
    173     }
    174     if(c >0xFFFF){
    175         ustr_ucat(dst, U16_LEAD(c), status);
    176         ustr_ucat(dst, U16_TRAIL(c), status);
    177     }else{
    178         ustr_ucat(dst, (UChar) c, status);
    179     }
    180 }
    181 U_CFUNC void
    182 ustr_uscat(struct UString *dst,
    183       const UChar* src,int len,
    184       UErrorCode *status)
    185 {
    186     if(U_FAILURE(*status))
    187         return;
    188 
    189     if(dst->fCapacity < (dst->fLength + len)) {
    190         ustr_resize(dst, ALLOCATION(dst->fLength + len), status);
    191         if(U_FAILURE(*status))
    192             return;
    193     }
    194 
    195     uprv_memcpy(dst->fChars + dst->fLength, src,
    196         sizeof(UChar) * len);
    197     dst->fLength += len;
    198     dst->fChars[dst->fLength] = 0x0000;
    199 }
    200 
    201 /* Destroys data in the string */
    202 static void
    203 ustr_resize(struct UString *s,
    204         int32_t len,
    205         UErrorCode *status)
    206 {
    207     if(U_FAILURE(*status))
    208         return;
    209 
    210     /* +1 for trailing 0x0000 */
    211     s->fChars = (UChar*) uprv_realloc(s->fChars, sizeof(UChar) * (len + 1));
    212     if(s->fChars == 0) {
    213         *status = U_MEMORY_ALLOCATION_ERROR;
    214         s->fLength = s->fCapacity = 0;
    215         return;
    216     }
    217 
    218     s->fCapacity = len;
    219 }
    220