/* ************************************************************************* * Copyright (C) 2016 and later: Unicode, Inc. and others. * License & terms of use: http://www.unicode.org/copyright.html#License ************************************************************************* ************************************************************************* * Copyright (C) 2014, International Business Machines * Corporation and others. All Rights Reserved. ************************************************************************* * file name: bitset.cpp * encoding: US-ASCII * tab size: 8 (not used) * indentation:4 * * created on: 2007jan15 * created by: Markus Scherer * * Idea for a "compiled", fast, read-only (immutable) version of a UnicodeSet * using a folded bit set consisting of a 1k-entry index table and a * compacted array of 64-bit words. * Uses a simple hash table for compaction. * Uses the original set for supplementary code points. */ #include "unicode/utypes.h" #include "unicont.h" #include "cmemory.h" // for UPRV_LENGTHOF /* * Hash table for up to 1k 64-bit words, for 1 bit per BMP code point. * Hashes 64-bit words and maps them to 16-bit integers which are * assigned in order of new incoming words for subsequent storage * in a contiguous array. */ struct BMPBitHash : public UObject { int64_t keys[0x800]; // 2k uint16_t values[0x800]; uint16_t reverse[0x400]; uint16_t count; const int32_t prime=1301; // Less than 2k. BMPBitHash() : count(0) { // Fill values[] with 0xffff. uprv_memset(values, 0xff, sizeof(values)); } /* * Map a key to an integer count. * Map at most 1k=0x400 different keys with this data structure. */ uint16_t map(int64_t key) { int32_t hash=(int32_t)(key>>55)&0x1ff; hash^=(int32_t)(key>>44)&0x7ff; hash^=(int32_t)(key>>33)&0x7ff; hash^=(int32_t)(key>>22)&0x7ff; hash^=(int32_t)(key>>11)&0x7ff; hash^=(int32_t)key&0x7ff; for(;;) { if(values[hash]==0xffff) { // Unused slot. keys[hash]=key; reverse[count]=hash; return values[hash]=count++; } else if(keys[hash]==key) { // Found a slot with this key. return values[hash]; } else { // Used slot with a different key, move to another slot. hash=(hash+prime)&0x7ff; } } } uint16_t countKeys() const { return count; } /* * Invert the hash map: Fill an array of length countKeys() with the keys * indexed by their mapped values. */ void invert(int64_t *k) const { uint16_t i; for(i=0; i<count; ++i) { k[i]=keys[reverse[i]]; } } }; class BitSet : public UObject, public UnicodeContainable { public: BitSet(const UnicodeSet &set, UErrorCode &errorCode) : bits(shortBits), restSet(set.clone()) { if(U_FAILURE(errorCode)) { return; } BMPBitHash *bitHash=new BMPBitHash; if(bitHash==NULL || restSet==NULL) { errorCode=U_MEMORY_ALLOCATION_ERROR; return; } UnicodeSetIterator iter(set); int64_t b; UChar32 start, end; int32_t prevIndex, i, j; b=0; // Not necessary but makes compilers happy. prevIndex=-1; for(;;) { if(iter.nextRange() && !iter.isString()) { start=iter.getCodepoint(); end=iter.getCodepointEnd(); } else { start=0x10000; } i=start>>6; if(prevIndex!=i) { // Finish the end of the previous range. if(prevIndex<0) { prevIndex=0; } else { index[prevIndex++]=bitHash->map(b); } // Fill all-zero entries between ranges. if(prevIndex<i) { uint16_t zero=bitHash->map(0); do { index[prevIndex++]=zero; } while(prevIndex<i); } b=0; } if(start>0xffff) { break; } b|=~((INT64_C(1)<<(start&0x3f))-1); j=end>>6; if(i<j) { // Set bits for the start of the range. index[i++]=bitHash->map(b); // Fill all-one entries inside the range. if(i<j) { uint16_t all=bitHash->map(INT64_C(0xffffffffffffffff)); do { index[i++]=all; } while(i<j); } b=INT64_C(0xffffffffffffffff); } /* i==j */ b&=(INT64_C(1)<<(end&0x3f))-1; prevIndex=j; } if(bitHash->countKeys()>UPRV_LENGTHOF(shortBits)) { bits=(int64_t *)uprv_malloc(bitHash->countKeys()*8); } if(bits!=NULL) { bitHash->invert(bits); } else { bits=shortBits; errorCode=U_MEMORY_ALLOCATION_ERROR; return; } latin1Set[0]=(uint32_t)bits[0]; latin1Set[1]=(uint32_t)(bits[0]>>32); latin1Set[2]=(uint32_t)bits[1]; latin1Set[3]=(uint32_t)(bits[1]>>32); latin1Set[4]=(uint32_t)bits[2]; latin1Set[5]=(uint32_t)(bits[2]>>32); latin1Set[6]=(uint32_t)bits[3]; latin1Set[7]=(uint32_t)(bits[3]>>32); restSet.remove(0, 0xffff); } ~BitSet() { if(bits!=shortBits) { uprv_free(bits); } delete restSet; } UBool contains(UChar32 c) const { if((uint32_t)c<=0xff) { return (UBool)((latin1Set[c>>5]&((uint32_t)1<<(c&0x1f)))!=0); } else if((uint32_t)c<0xffff) { return (UBool)((bits[c>>6]&(INT64_C(1)<<(c&0x3f)))!=0); } else { return restSet->contains(c); } } private: uint16_t index[0x400]; int64_t shortBits[32]; int64_t *bits; uint32_t latin1Bits[8]; UnicodeSet *restSet; };