1 /* 2 * PROJECT: ReactOS Application compatibility module 3 * LICENSE: GPL-2.0-or-later (https://spdx.org/licenses/GPL-2.0-or-later) 4 * PURPOSE: Shim database string table builder 5 * COPYRIGHT: Copyright 2016-2019 Mark Jansen (mark.jansen@reactos.org) 6 */ 7 8 #if !defined(SDBWRITE_HOSTTOOL) 9 #define WIN32_NO_STATUS 10 #include "windows.h" 11 #include <appcompat/sdbtypes.h> 12 #include "sdbpapi.h" 13 #else /* !defined(SDBWRITE_HOSTTOOL) */ 14 #include <typedefs.h> 15 #include <guiddef.h> 16 #include "sdbtypes.h" 17 #include "sdbpapi.h" 18 #endif /* !defined(SDBWRITE_HOSTTOOL) */ 19 20 #include "sdbstringtable.h" 21 22 #if !defined(offsetof) 23 #if defined(__GNUC__) 24 #define offsetof(TYPE, MEMBER) __builtin_offsetof (TYPE, MEMBER) 25 #else 26 #define offsetof(TYPE, MEMBER) ((size_t)&(((TYPE *)0)->MEMBER)) 27 #endif 28 #endif // !defined(offsetof) 29 30 #define DEFAULT_TABLE_SIZE 0x100 31 32 typedef struct SdbHashEntry 33 { 34 struct SdbHashEntry* Next; 35 TAGID Tagid; 36 WCHAR Name[1]; 37 } SdbHashEntry; 38 39 struct SdbStringHashTable 40 { 41 DWORD Size; 42 struct SdbHashEntry** Entries; 43 }; 44 45 46 static struct SdbStringHashTable* HashCreate(void) 47 { 48 struct SdbStringHashTable* tab = SdbAlloc(sizeof(*tab)); 49 if (!tab) 50 { 51 SHIM_ERR("Failed to allocate 8 bytes.\r\n"); 52 return tab; 53 } 54 tab->Size = DEFAULT_TABLE_SIZE; 55 tab->Entries = SdbAlloc(tab->Size * sizeof(*tab->Entries)); 56 return tab; 57 } 58 59 60 void SdbpTableDestroy(struct SdbStringHashTable** pTable) 61 { 62 struct SdbStringHashTable* table = *pTable; 63 struct SdbHashEntry* entry, *next; 64 DWORD n, depth = 0, once = 1; 65 66 *pTable = NULL; 67 for (n = 0; n < table->Size; ++n) 68 { 69 depth = 0; 70 entry = next = table->Entries[n]; 71 while (entry) 72 { 73 next = entry->Next; 74 SdbFree(entry); 75 entry = next; 76 depth++; 77 } 78 if (once && depth > 3) 79 { 80 // warn 81 once = 0; 82 } 83 } 84 SdbFree(table->Entries); 85 SdbFree(table); 86 } 87 88 /* Based on RtlHashUnicodeString */ 89 static DWORD StringHash(const WCHAR* str) 90 { 91 DWORD hash = 0; 92 for (; *str; str++) 93 { 94 hash = ((65599 * hash) + (ULONG)(*str)); 95 } 96 return hash; 97 } 98 99 int Sdbwcscmp(const WCHAR* s1, const WCHAR* s2) 100 { 101 while (*s1 == *s2) 102 { 103 if (*s1 == 0) 104 return 0; 105 s1++; 106 s2++; 107 } 108 return *s1 - *s2; 109 } 110 111 112 // implementation taken from reactos/sdk/lib/crt/string/wcs.c 113 INT Sdbwcscpy(WCHAR* wcDest, size_t numElement, const WCHAR *wcSrc) 114 { 115 size_t size = 0; 116 if(!wcDest || !numElement) 117 return 22; /* EINVAL */ 118 119 wcDest[0] = 0; 120 121 if(!wcSrc) 122 return 22; /* EINVAL */ 123 124 size = SdbpStrlen(wcSrc) + 1; 125 126 if(size > numElement) 127 return 34; /* ERANGE */ 128 129 memcpy(wcDest, wcSrc, size * sizeof(WCHAR)); 130 131 return 0; 132 } 133 134 static struct SdbHashEntry** TableFindPtr(struct SdbStringHashTable* table, const WCHAR* str) 135 { 136 DWORD hash = StringHash(str); 137 struct SdbHashEntry** entry = &table->Entries[hash % table->Size]; 138 while (*entry) 139 { 140 if (!Sdbwcscmp((*entry)->Name, str)) 141 return entry; 142 entry = &(*entry)->Next; 143 } 144 return entry; 145 } 146 147 static BOOL HashAddString(struct SdbStringHashTable* table, struct SdbHashEntry** position, const WCHAR* str, TAGID tagid) 148 { 149 struct SdbHashEntry* entry; 150 SIZE_T size, len; 151 152 if (!position) 153 position = TableFindPtr(table, str); 154 155 len = SdbpStrlen(str) + 1; 156 size = offsetof(struct SdbHashEntry, Name[len]); 157 entry = (*position) = SdbAlloc(size); 158 if (!entry) 159 { 160 SHIM_ERR("Failed to allocate %u bytes.", size); 161 return FALSE; 162 } 163 entry->Tagid = tagid; 164 Sdbwcscpy(entry->Name, len, str); 165 return TRUE; 166 } 167 168 169 BOOL SdbpAddStringToTable(struct SdbStringHashTable** table, const WCHAR* str, TAGID* tagid) 170 { 171 struct SdbHashEntry** entry; 172 173 if (!*table) 174 { 175 *table = HashCreate(); 176 if (!*table) 177 { 178 SHIM_ERR("Error creating hash table\n"); 179 return FALSE; 180 } 181 } 182 183 entry = TableFindPtr(*table, str); 184 if (*entry) 185 { 186 *tagid = (*entry)->Tagid; 187 return FALSE; 188 } 189 return HashAddString(*table, entry, str, *tagid); 190 } 191 192