|  | /* | 
|  | ******************************************************************************* | 
|  | * | 
|  | *   Copyright (C) 2004-2014, International Business Machines | 
|  | *   Corporation and others.  All Rights Reserved. | 
|  | * | 
|  | ******************************************************************************* | 
|  | *   file name:  ubidi_props.c | 
|  | *   encoding:   US-ASCII | 
|  | *   tab size:   8 (not used) | 
|  | *   indentation:4 | 
|  | * | 
|  | *   created on: 2004dec30 | 
|  | *   created by: Markus W. Scherer | 
|  | * | 
|  | *   Low-level Unicode bidi/shaping properties access. | 
|  | */ | 
|  |  | 
|  | #include "unicode/utypes.h" | 
|  | #include "unicode/uset.h" | 
|  | #include "unicode/udata.h" /* UDataInfo */ | 
|  | #include "ucmndata.h" /* DataHeader */ | 
|  | #include "udatamem.h" | 
|  | #include "uassert.h" | 
|  | #include "cmemory.h" | 
|  | #include "utrie2.h" | 
|  | #include "ubidi_props.h" | 
|  | #include "ucln_cmn.h" | 
|  |  | 
|  | struct UBiDiProps { | 
|  | UDataMemory *mem; | 
|  | const int32_t *indexes; | 
|  | const uint32_t *mirrors; | 
|  | const uint8_t *jgArray; | 
|  | const uint8_t *jgArray2; | 
|  |  | 
|  | UTrie2 trie; | 
|  | uint8_t formatVersion[4]; | 
|  | }; | 
|  |  | 
|  | /* ubidi_props_data.h is machine-generated by genbidi --csource */ | 
|  | #define INCLUDED_FROM_UBIDI_PROPS_C | 
|  | #include "ubidi_props_data.h" | 
|  |  | 
|  | /* UBiDiProps singleton ----------------------------------------------------- */ | 
|  |  | 
|  | U_CFUNC const UBiDiProps * | 
|  | ubidi_getSingleton() { | 
|  | return &ubidi_props_singleton; | 
|  | } | 
|  |  | 
|  | /* set of property starts for UnicodeSet ------------------------------------ */ | 
|  |  | 
|  | static UBool U_CALLCONV | 
|  | _enumPropertyStartsRange(const void *context, UChar32 start, UChar32 end, uint32_t value) { | 
|  | /* add the start code point to the USet */ | 
|  | const USetAdder *sa=(const USetAdder *)context; | 
|  | sa->add(sa->set, start); | 
|  | return TRUE; | 
|  | } | 
|  |  | 
|  | U_CFUNC void | 
|  | ubidi_addPropertyStarts(const UBiDiProps *bdp, const USetAdder *sa, UErrorCode *pErrorCode) { | 
|  | int32_t i, length; | 
|  | UChar32 c, start, limit; | 
|  |  | 
|  | const uint8_t *jgArray; | 
|  | uint8_t prev, jg; | 
|  |  | 
|  | if(U_FAILURE(*pErrorCode)) { | 
|  | return; | 
|  | } | 
|  |  | 
|  | /* add the start code point of each same-value range of the trie */ | 
|  | utrie2_enum(&bdp->trie, NULL, _enumPropertyStartsRange, sa); | 
|  |  | 
|  | /* add the code points from the bidi mirroring table */ | 
|  | length=bdp->indexes[UBIDI_IX_MIRROR_LENGTH]; | 
|  | for(i=0; i<length; ++i) { | 
|  | c=UBIDI_GET_MIRROR_CODE_POINT(bdp->mirrors[i]); | 
|  | sa->addRange(sa->set, c, c+1); | 
|  | } | 
|  |  | 
|  | /* add the code points from the Joining_Group array where the value changes */ | 
|  | start=bdp->indexes[UBIDI_IX_JG_START]; | 
|  | limit=bdp->indexes[UBIDI_IX_JG_LIMIT]; | 
|  | jgArray=bdp->jgArray; | 
|  | for(;;) { | 
|  | prev=0; | 
|  | while(start<limit) { | 
|  | jg=*jgArray++; | 
|  | if(jg!=prev) { | 
|  | sa->add(sa->set, start); | 
|  | prev=jg; | 
|  | } | 
|  | ++start; | 
|  | } | 
|  | if(prev!=0) { | 
|  | /* add the limit code point if the last value was not 0 (it is now start==limit) */ | 
|  | sa->add(sa->set, limit); | 
|  | } | 
|  | if(limit==bdp->indexes[UBIDI_IX_JG_LIMIT]) { | 
|  | /* switch to the second Joining_Group range */ | 
|  | start=bdp->indexes[UBIDI_IX_JG_START2]; | 
|  | limit=bdp->indexes[UBIDI_IX_JG_LIMIT2]; | 
|  | jgArray=bdp->jgArray2; | 
|  | } else { | 
|  | break; | 
|  | } | 
|  | } | 
|  |  | 
|  | /* add code points with hardcoded properties, plus the ones following them */ | 
|  |  | 
|  | /* (none right now) */ | 
|  | } | 
|  |  | 
|  | /* property access functions ------------------------------------------------ */ | 
|  |  | 
|  | U_CFUNC int32_t | 
|  | ubidi_getMaxValue(const UBiDiProps *bdp, UProperty which) { | 
|  | int32_t max; | 
|  |  | 
|  | if(bdp==NULL) { | 
|  | return -1; | 
|  | } | 
|  |  | 
|  | max=bdp->indexes[UBIDI_MAX_VALUES_INDEX]; | 
|  | switch(which) { | 
|  | case UCHAR_BIDI_CLASS: | 
|  | return (max&UBIDI_CLASS_MASK); | 
|  | case UCHAR_JOINING_GROUP: | 
|  | return (max&UBIDI_MAX_JG_MASK)>>UBIDI_MAX_JG_SHIFT; | 
|  | case UCHAR_JOINING_TYPE: | 
|  | return (max&UBIDI_JT_MASK)>>UBIDI_JT_SHIFT; | 
|  | case UCHAR_BIDI_PAIRED_BRACKET_TYPE: | 
|  | return (max&UBIDI_BPT_MASK)>>UBIDI_BPT_SHIFT; | 
|  | default: | 
|  | return -1; /* undefined */ | 
|  | } | 
|  | } | 
|  |  | 
|  | U_CAPI UCharDirection | 
|  | ubidi_getClass(const UBiDiProps *bdp, UChar32 c) { | 
|  | uint16_t props=UTRIE2_GET16(&bdp->trie, c); | 
|  | return (UCharDirection)UBIDI_GET_CLASS(props); | 
|  | } | 
|  |  | 
|  | U_CFUNC UBool | 
|  | ubidi_isMirrored(const UBiDiProps *bdp, UChar32 c) { | 
|  | uint16_t props=UTRIE2_GET16(&bdp->trie, c); | 
|  | return (UBool)UBIDI_GET_FLAG(props, UBIDI_IS_MIRRORED_SHIFT); | 
|  | } | 
|  |  | 
|  | static UChar32 | 
|  | getMirror(const UBiDiProps *bdp, UChar32 c, uint16_t props) { | 
|  | int32_t delta=UBIDI_GET_MIRROR_DELTA(props); | 
|  | if(delta!=UBIDI_ESC_MIRROR_DELTA) { | 
|  | return c+delta; | 
|  | } else { | 
|  | /* look for mirror code point in the mirrors[] table */ | 
|  | const uint32_t *mirrors; | 
|  | uint32_t m; | 
|  | int32_t i, length; | 
|  | UChar32 c2; | 
|  |  | 
|  | mirrors=bdp->mirrors; | 
|  | length=bdp->indexes[UBIDI_IX_MIRROR_LENGTH]; | 
|  |  | 
|  | /* linear search */ | 
|  | for(i=0; i<length; ++i) { | 
|  | m=mirrors[i]; | 
|  | c2=UBIDI_GET_MIRROR_CODE_POINT(m); | 
|  | if(c==c2) { | 
|  | /* found c, return its mirror code point using the index in m */ | 
|  | return UBIDI_GET_MIRROR_CODE_POINT(mirrors[UBIDI_GET_MIRROR_INDEX(m)]); | 
|  | } else if(c<c2) { | 
|  | break; | 
|  | } | 
|  | } | 
|  |  | 
|  | /* c not found, return it itself */ | 
|  | return c; | 
|  | } | 
|  | } | 
|  |  | 
|  | U_CFUNC UChar32 | 
|  | ubidi_getMirror(const UBiDiProps *bdp, UChar32 c) { | 
|  | uint16_t props=UTRIE2_GET16(&bdp->trie, c); | 
|  | return getMirror(bdp, c, props); | 
|  | } | 
|  |  | 
|  | U_CFUNC UBool | 
|  | ubidi_isBidiControl(const UBiDiProps *bdp, UChar32 c) { | 
|  | uint16_t props=UTRIE2_GET16(&bdp->trie, c); | 
|  | return (UBool)UBIDI_GET_FLAG(props, UBIDI_BIDI_CONTROL_SHIFT); | 
|  | } | 
|  |  | 
|  | U_CFUNC UBool | 
|  | ubidi_isJoinControl(const UBiDiProps *bdp, UChar32 c) { | 
|  | uint16_t props=UTRIE2_GET16(&bdp->trie, c); | 
|  | return (UBool)UBIDI_GET_FLAG(props, UBIDI_JOIN_CONTROL_SHIFT); | 
|  | } | 
|  |  | 
|  | U_CFUNC UJoiningType | 
|  | ubidi_getJoiningType(const UBiDiProps *bdp, UChar32 c) { | 
|  | uint16_t props=UTRIE2_GET16(&bdp->trie, c); | 
|  | return (UJoiningType)((props&UBIDI_JT_MASK)>>UBIDI_JT_SHIFT); | 
|  | } | 
|  |  | 
|  | U_CFUNC UJoiningGroup | 
|  | ubidi_getJoiningGroup(const UBiDiProps *bdp, UChar32 c) { | 
|  | UChar32 start, limit; | 
|  |  | 
|  | start=bdp->indexes[UBIDI_IX_JG_START]; | 
|  | limit=bdp->indexes[UBIDI_IX_JG_LIMIT]; | 
|  | if(start<=c && c<limit) { | 
|  | return (UJoiningGroup)bdp->jgArray[c-start]; | 
|  | } | 
|  | start=bdp->indexes[UBIDI_IX_JG_START2]; | 
|  | limit=bdp->indexes[UBIDI_IX_JG_LIMIT2]; | 
|  | if(start<=c && c<limit) { | 
|  | return (UJoiningGroup)bdp->jgArray2[c-start]; | 
|  | } | 
|  | return U_JG_NO_JOINING_GROUP; | 
|  | } | 
|  |  | 
|  | U_CFUNC UBidiPairedBracketType | 
|  | ubidi_getPairedBracketType(const UBiDiProps *bdp, UChar32 c) { | 
|  | uint16_t props=UTRIE2_GET16(&bdp->trie, c); | 
|  | return (UBidiPairedBracketType)((props&UBIDI_BPT_MASK)>>UBIDI_BPT_SHIFT); | 
|  | } | 
|  |  | 
|  | U_CFUNC UChar32 | 
|  | ubidi_getPairedBracket(const UBiDiProps *bdp, UChar32 c) { | 
|  | uint16_t props=UTRIE2_GET16(&bdp->trie, c); | 
|  | if((props&UBIDI_BPT_MASK)==0) { | 
|  | return c; | 
|  | } else { | 
|  | return getMirror(bdp, c, props); | 
|  | } | 
|  | } | 
|  |  | 
|  | /* public API (see uchar.h) ------------------------------------------------- */ | 
|  |  | 
|  | U_CFUNC UCharDirection | 
|  | u_charDirection(UChar32 c) { | 
|  | return ubidi_getClass(&ubidi_props_singleton, c); | 
|  | } | 
|  |  | 
|  | U_CFUNC UBool | 
|  | u_isMirrored(UChar32 c) { | 
|  | return ubidi_isMirrored(&ubidi_props_singleton, c); | 
|  | } | 
|  |  | 
|  | U_CFUNC UChar32 | 
|  | u_charMirror(UChar32 c) { | 
|  | return ubidi_getMirror(&ubidi_props_singleton, c); | 
|  | } | 
|  |  | 
|  | U_STABLE UChar32 U_EXPORT2 | 
|  | u_getBidiPairedBracket(UChar32 c) { | 
|  | return ubidi_getPairedBracket(&ubidi_props_singleton, c); | 
|  | } |