Mike McCormack | f5e1c13 | 2004-03-17 20:49:59 +0000 | [diff] [blame] | 1 | /* |
Rob Shearman | e261d60 | 2007-04-23 08:18:27 +0100 | [diff] [blame] | 2 | * String Table Functions |
Mike McCormack | f5e1c13 | 2004-03-17 20:49:59 +0000 | [diff] [blame] | 3 | * |
Mike McCormack | 068b4ec | 2004-03-19 01:16:36 +0000 | [diff] [blame] | 4 | * Copyright 2002-2004, Mike McCormack for CodeWeavers |
Rob Shearman | e261d60 | 2007-04-23 08:18:27 +0100 | [diff] [blame] | 5 | * Copyright 2007 Robert Shearman for CodeWeavers |
Mike McCormack | f5e1c13 | 2004-03-17 20:49:59 +0000 | [diff] [blame] | 6 | * |
| 7 | * This library is free software; you can redistribute it and/or |
| 8 | * modify it under the terms of the GNU Lesser General Public |
| 9 | * License as published by the Free Software Foundation; either |
| 10 | * version 2.1 of the License, or (at your option) any later version. |
| 11 | * |
| 12 | * This library is distributed in the hope that it will be useful, |
| 13 | * but WITHOUT ANY WARRANTY; without even the implied warranty of |
| 14 | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU |
| 15 | * Lesser General Public License for more details. |
| 16 | * |
| 17 | * You should have received a copy of the GNU Lesser General Public |
| 18 | * License along with this library; if not, write to the Free Software |
Jonathan Ernst | 360a3f9 | 2006-05-18 14:49:52 +0200 | [diff] [blame] | 19 | * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA |
Mike McCormack | f5e1c13 | 2004-03-17 20:49:59 +0000 | [diff] [blame] | 20 | */ |
| 21 | |
Rob Shearman | 37b11ba | 2007-04-23 08:17:40 +0100 | [diff] [blame] | 22 | #define COBJMACROS |
| 23 | |
Mike McCormack | f5e1c13 | 2004-03-17 20:49:59 +0000 | [diff] [blame] | 24 | #include <stdarg.h> |
| 25 | #include <assert.h> |
| 26 | |
| 27 | #include "windef.h" |
| 28 | #include "winbase.h" |
| 29 | #include "winerror.h" |
| 30 | #include "wine/debug.h" |
Mike McCormack | ab519f2 | 2004-06-30 18:18:27 +0000 | [diff] [blame] | 31 | #include "wine/unicode.h" |
Mike McCormack | f5e1c13 | 2004-03-17 20:49:59 +0000 | [diff] [blame] | 32 | #include "msi.h" |
| 33 | #include "msiquery.h" |
| 34 | #include "objbase.h" |
| 35 | #include "objidl.h" |
| 36 | #include "msipriv.h" |
| 37 | #include "winnls.h" |
| 38 | |
| 39 | #include "query.h" |
| 40 | |
Mike McCormack | 50684c1 | 2005-11-02 14:24:21 +0000 | [diff] [blame] | 41 | WINE_DEFAULT_DEBUG_CHANNEL(msidb); |
Mike McCormack | f5e1c13 | 2004-03-17 20:49:59 +0000 | [diff] [blame] | 42 | |
Mike McCormack | 4f9cae8 | 2006-09-08 16:20:56 +0900 | [diff] [blame] | 43 | #define HASH_SIZE 0x101 |
James Hawkins | a05613a | 2007-06-07 16:41:51 -0700 | [diff] [blame] | 44 | #define LONG_STR_BYTES 3 |
Mike McCormack | f537eb2 | 2006-04-03 18:48:52 +0900 | [diff] [blame] | 45 | |
Mike McCormack | f5e1c13 | 2004-03-17 20:49:59 +0000 | [diff] [blame] | 46 | typedef struct _msistring |
| 47 | { |
Mike McCormack | f537eb2 | 2006-04-03 18:48:52 +0900 | [diff] [blame] | 48 | int hash_next; |
Rob Shearman | 2e3289c | 2007-04-23 08:17:00 +0100 | [diff] [blame] | 49 | UINT persistent_refcount; |
| 50 | UINT nonpersistent_refcount; |
Mike McCormack | ab519f2 | 2004-06-30 18:18:27 +0000 | [diff] [blame] | 51 | LPWSTR str; |
Mike McCormack | f5e1c13 | 2004-03-17 20:49:59 +0000 | [diff] [blame] | 52 | } msistring; |
| 53 | |
| 54 | struct string_table |
| 55 | { |
Mike McCormack | abd259f | 2004-06-30 18:24:33 +0000 | [diff] [blame] | 56 | UINT maxcount; /* the number of strings */ |
Mike McCormack | f5e1c13 | 2004-03-17 20:49:59 +0000 | [diff] [blame] | 57 | UINT freeslot; |
Mike McCormack | c0523aa | 2004-06-26 00:18:36 +0000 | [diff] [blame] | 58 | UINT codepage; |
Mike McCormack | f537eb2 | 2006-04-03 18:48:52 +0900 | [diff] [blame] | 59 | int hash[HASH_SIZE]; |
Mike McCormack | f5e1c13 | 2004-03-17 20:49:59 +0000 | [diff] [blame] | 60 | msistring *strings; /* an array of strings (in the tree) */ |
| 61 | }; |
| 62 | |
Mike McCormack | 4604e66 | 2004-08-06 17:30:20 +0000 | [diff] [blame] | 63 | static UINT msistring_makehash( const WCHAR *str ) |
Mike McCormack | f5e1c13 | 2004-03-17 20:49:59 +0000 | [diff] [blame] | 64 | { |
Mike McCormack | 4604e66 | 2004-08-06 17:30:20 +0000 | [diff] [blame] | 65 | UINT hash = 0; |
Mike McCormack | f5e1c13 | 2004-03-17 20:49:59 +0000 | [diff] [blame] | 66 | |
Aric Stewart | 8d81d67 | 2004-07-29 23:59:15 +0000 | [diff] [blame] | 67 | if (str==NULL) |
| 68 | return hash; |
| 69 | |
Mike McCormack | f5e1c13 | 2004-03-17 20:49:59 +0000 | [diff] [blame] | 70 | while( *str ) |
| 71 | { |
| 72 | hash ^= *str++; |
| 73 | hash *= 53; |
Aric Stewart | 8d81d67 | 2004-07-29 23:59:15 +0000 | [diff] [blame] | 74 | hash = (hash<<5) | (hash>>27); |
Mike McCormack | f5e1c13 | 2004-03-17 20:49:59 +0000 | [diff] [blame] | 75 | } |
Mike McCormack | f537eb2 | 2006-04-03 18:48:52 +0900 | [diff] [blame] | 76 | return hash % HASH_SIZE; |
Mike McCormack | f5e1c13 | 2004-03-17 20:49:59 +0000 | [diff] [blame] | 77 | } |
| 78 | |
Rob Shearman | e261d60 | 2007-04-23 08:18:27 +0100 | [diff] [blame] | 79 | static string_table *init_stringtable( int entries, UINT codepage ) |
Mike McCormack | f5e1c13 | 2004-03-17 20:49:59 +0000 | [diff] [blame] | 80 | { |
| 81 | string_table *st; |
Mike McCormack | f537eb2 | 2006-04-03 18:48:52 +0900 | [diff] [blame] | 82 | int i; |
Mike McCormack | f5e1c13 | 2004-03-17 20:49:59 +0000 | [diff] [blame] | 83 | |
Mike McCormack | 8dc28d5 | 2005-09-20 11:57:19 +0000 | [diff] [blame] | 84 | st = msi_alloc( sizeof (string_table) ); |
Mike McCormack | f5e1c13 | 2004-03-17 20:49:59 +0000 | [diff] [blame] | 85 | if( !st ) |
| 86 | return NULL; |
Mike McCormack | de71dbb | 2005-09-26 09:54:56 +0000 | [diff] [blame] | 87 | if( entries < 1 ) |
| 88 | entries = 1; |
Mike McCormack | 8dc28d5 | 2005-09-20 11:57:19 +0000 | [diff] [blame] | 89 | st->strings = msi_alloc_zero( sizeof (msistring) * entries ); |
Mike McCormack | ed9745b | 2006-04-07 13:30:40 +0900 | [diff] [blame] | 90 | if( !st->strings ) |
Mike McCormack | f5e1c13 | 2004-03-17 20:49:59 +0000 | [diff] [blame] | 91 | { |
Mike McCormack | 8dc28d5 | 2005-09-20 11:57:19 +0000 | [diff] [blame] | 92 | msi_free( st ); |
Mike McCormack | f5e1c13 | 2004-03-17 20:49:59 +0000 | [diff] [blame] | 93 | return NULL; |
| 94 | } |
Mike McCormack | abd259f | 2004-06-30 18:24:33 +0000 | [diff] [blame] | 95 | st->maxcount = entries; |
Mike McCormack | 068b4ec | 2004-03-19 01:16:36 +0000 | [diff] [blame] | 96 | st->freeslot = 1; |
Mike McCormack | c0523aa | 2004-06-26 00:18:36 +0000 | [diff] [blame] | 97 | st->codepage = codepage; |
Mike McCormack | f5e1c13 | 2004-03-17 20:49:59 +0000 | [diff] [blame] | 98 | |
Mike McCormack | f537eb2 | 2006-04-03 18:48:52 +0900 | [diff] [blame] | 99 | for( i=0; i<HASH_SIZE; i++ ) |
| 100 | st->hash[i] = -1; |
| 101 | |
Mike McCormack | f5e1c13 | 2004-03-17 20:49:59 +0000 | [diff] [blame] | 102 | return st; |
| 103 | } |
| 104 | |
| 105 | VOID msi_destroy_stringtable( string_table *st ) |
| 106 | { |
| 107 | UINT i; |
| 108 | |
Mike McCormack | abd259f | 2004-06-30 18:24:33 +0000 | [diff] [blame] | 109 | for( i=0; i<st->maxcount; i++ ) |
Mike McCormack | f5e1c13 | 2004-03-17 20:49:59 +0000 | [diff] [blame] | 110 | { |
Rob Shearman | 2e3289c | 2007-04-23 08:17:00 +0100 | [diff] [blame] | 111 | if( st->strings[i].persistent_refcount || |
| 112 | st->strings[i].nonpersistent_refcount ) |
Mike McCormack | 8dc28d5 | 2005-09-20 11:57:19 +0000 | [diff] [blame] | 113 | msi_free( st->strings[i].str ); |
Mike McCormack | f5e1c13 | 2004-03-17 20:49:59 +0000 | [diff] [blame] | 114 | } |
Mike McCormack | 8dc28d5 | 2005-09-20 11:57:19 +0000 | [diff] [blame] | 115 | msi_free( st->strings ); |
| 116 | msi_free( st ); |
Mike McCormack | f5e1c13 | 2004-03-17 20:49:59 +0000 | [diff] [blame] | 117 | } |
| 118 | |
| 119 | static int st_find_free_entry( string_table *st ) |
| 120 | { |
Mike McCormack | 4604e66 | 2004-08-06 17:30:20 +0000 | [diff] [blame] | 121 | UINT i, sz; |
Mike McCormack | 068b4ec | 2004-03-19 01:16:36 +0000 | [diff] [blame] | 122 | msistring *p; |
Mike McCormack | f5e1c13 | 2004-03-17 20:49:59 +0000 | [diff] [blame] | 123 | |
Mike McCormack | ab519f2 | 2004-06-30 18:18:27 +0000 | [diff] [blame] | 124 | TRACE("%p\n", st); |
| 125 | |
Mike McCormack | abd259f | 2004-06-30 18:24:33 +0000 | [diff] [blame] | 126 | if( st->freeslot ) |
| 127 | { |
| 128 | for( i = st->freeslot; i < st->maxcount; i++ ) |
Rob Shearman | 2e3289c | 2007-04-23 08:17:00 +0100 | [diff] [blame] | 129 | if( !st->strings[i].persistent_refcount && |
| 130 | !st->strings[i].nonpersistent_refcount ) |
Mike McCormack | abd259f | 2004-06-30 18:24:33 +0000 | [diff] [blame] | 131 | return i; |
| 132 | } |
| 133 | for( i = 1; i < st->maxcount; i++ ) |
Rob Shearman | 2e3289c | 2007-04-23 08:17:00 +0100 | [diff] [blame] | 134 | if( !st->strings[i].persistent_refcount && |
| 135 | !st->strings[i].nonpersistent_refcount ) |
Mike McCormack | f5e1c13 | 2004-03-17 20:49:59 +0000 | [diff] [blame] | 136 | return i; |
| 137 | |
Mike McCormack | 068b4ec | 2004-03-19 01:16:36 +0000 | [diff] [blame] | 138 | /* dynamically resize */ |
Mike McCormack | abd259f | 2004-06-30 18:24:33 +0000 | [diff] [blame] | 139 | sz = st->maxcount + 1 + st->maxcount/2; |
Mike McCormack | 8dc28d5 | 2005-09-20 11:57:19 +0000 | [diff] [blame] | 140 | p = msi_realloc_zero( st->strings, sz*sizeof(msistring) ); |
Mike McCormack | 068b4ec | 2004-03-19 01:16:36 +0000 | [diff] [blame] | 141 | if( !p ) |
| 142 | return -1; |
| 143 | st->strings = p; |
Mike McCormack | abd259f | 2004-06-30 18:24:33 +0000 | [diff] [blame] | 144 | st->freeslot = st->maxcount; |
| 145 | st->maxcount = sz; |
Rob Shearman | 2e3289c | 2007-04-23 08:17:00 +0100 | [diff] [blame] | 146 | if( st->strings[st->freeslot].persistent_refcount || |
| 147 | st->strings[st->freeslot].nonpersistent_refcount ) |
Mike McCormack | 068b4ec | 2004-03-19 01:16:36 +0000 | [diff] [blame] | 148 | ERR("oops. expected freeslot to be free...\n"); |
| 149 | return st->freeslot; |
Mike McCormack | f5e1c13 | 2004-03-17 20:49:59 +0000 | [diff] [blame] | 150 | } |
| 151 | |
Rob Shearman | 2e3289c | 2007-04-23 08:17:00 +0100 | [diff] [blame] | 152 | static void set_st_entry( string_table *st, UINT n, LPWSTR str, UINT refcount, enum StringPersistence persistence ) |
Mike McCormack | f5e1c13 | 2004-03-17 20:49:59 +0000 | [diff] [blame] | 153 | { |
Mike McCormack | f537eb2 | 2006-04-03 18:48:52 +0900 | [diff] [blame] | 154 | UINT hash = msistring_makehash( str ); |
| 155 | |
Rob Shearman | 2e3289c | 2007-04-23 08:17:00 +0100 | [diff] [blame] | 156 | if (persistence == StringPersistent) |
| 157 | { |
| 158 | st->strings[n].persistent_refcount = refcount; |
| 159 | st->strings[n].nonpersistent_refcount = 0; |
| 160 | } |
| 161 | else |
| 162 | { |
| 163 | st->strings[n].persistent_refcount = 0; |
| 164 | st->strings[n].nonpersistent_refcount = refcount; |
| 165 | } |
| 166 | |
Mike McCormack | f537eb2 | 2006-04-03 18:48:52 +0900 | [diff] [blame] | 167 | st->strings[n].str = str; |
| 168 | |
| 169 | st->strings[n].hash_next = st->hash[hash]; |
| 170 | st->hash[hash] = n; |
| 171 | |
| 172 | if( n < st->maxcount ) |
| 173 | st->freeslot = n + 1; |
Mike McCormack | f5e1c13 | 2004-03-17 20:49:59 +0000 | [diff] [blame] | 174 | } |
| 175 | |
Rob Shearman | e261d60 | 2007-04-23 08:18:27 +0100 | [diff] [blame] | 176 | static int msi_addstring( string_table *st, UINT n, const CHAR *data, int len, UINT refcount, enum StringPersistence persistence ) |
Mike McCormack | f5e1c13 | 2004-03-17 20:49:59 +0000 | [diff] [blame] | 177 | { |
Mike McCormack | f537eb2 | 2006-04-03 18:48:52 +0900 | [diff] [blame] | 178 | LPWSTR str; |
Mike McCormack | ab519f2 | 2004-06-30 18:18:27 +0000 | [diff] [blame] | 179 | int sz; |
| 180 | |
Mike McCormack | de8674e | 2004-06-30 18:42:02 +0000 | [diff] [blame] | 181 | if( !data ) |
| 182 | return 0; |
Mike McCormack | 068b4ec | 2004-03-19 01:16:36 +0000 | [diff] [blame] | 183 | if( !data[0] ) |
| 184 | return 0; |
| 185 | if( n > 0 ) |
| 186 | { |
Rob Shearman | 2e3289c | 2007-04-23 08:17:00 +0100 | [diff] [blame] | 187 | if( st->strings[n].persistent_refcount || |
| 188 | st->strings[n].nonpersistent_refcount ) |
Mike McCormack | 068b4ec | 2004-03-19 01:16:36 +0000 | [diff] [blame] | 189 | return -1; |
| 190 | } |
| 191 | else |
| 192 | { |
| 193 | if( ERROR_SUCCESS == msi_string2idA( st, data, &n ) ) |
| 194 | { |
Rob Shearman | 2e3289c | 2007-04-23 08:17:00 +0100 | [diff] [blame] | 195 | if (persistence == StringPersistent) |
| 196 | st->strings[n].persistent_refcount += refcount; |
| 197 | else |
| 198 | st->strings[n].nonpersistent_refcount += refcount; |
Mike McCormack | 068b4ec | 2004-03-19 01:16:36 +0000 | [diff] [blame] | 199 | return n; |
| 200 | } |
| 201 | n = st_find_free_entry( st ); |
| 202 | if( n < 0 ) |
| 203 | return -1; |
| 204 | } |
Mike McCormack | f5e1c13 | 2004-03-17 20:49:59 +0000 | [diff] [blame] | 205 | |
Mike McCormack | abd259f | 2004-06-30 18:24:33 +0000 | [diff] [blame] | 206 | if( n < 1 ) |
| 207 | { |
| 208 | ERR("invalid index adding %s (%d)\n", debugstr_a( data ), n ); |
| 209 | return -1; |
| 210 | } |
| 211 | |
Mike McCormack | f5e1c13 | 2004-03-17 20:49:59 +0000 | [diff] [blame] | 212 | /* allocate a new string */ |
| 213 | if( len < 0 ) |
Mike McCormack | ab519f2 | 2004-06-30 18:18:27 +0000 | [diff] [blame] | 214 | len = strlen(data); |
| 215 | sz = MultiByteToWideChar( st->codepage, 0, data, len, NULL, 0 ); |
Mike McCormack | f537eb2 | 2006-04-03 18:48:52 +0900 | [diff] [blame] | 216 | str = msi_alloc( (sz+1)*sizeof(WCHAR) ); |
| 217 | if( !str ) |
Mike McCormack | f5e1c13 | 2004-03-17 20:49:59 +0000 | [diff] [blame] | 218 | return -1; |
Mike McCormack | f537eb2 | 2006-04-03 18:48:52 +0900 | [diff] [blame] | 219 | MultiByteToWideChar( st->codepage, 0, data, len, str, sz ); |
| 220 | str[sz] = 0; |
Mike McCormack | f5e1c13 | 2004-03-17 20:49:59 +0000 | [diff] [blame] | 221 | |
Rob Shearman | 2e3289c | 2007-04-23 08:17:00 +0100 | [diff] [blame] | 222 | set_st_entry( st, n, str, refcount, persistence ); |
Mike McCormack | f5e1c13 | 2004-03-17 20:49:59 +0000 | [diff] [blame] | 223 | |
| 224 | return n; |
| 225 | } |
| 226 | |
Rob Shearman | 2e3289c | 2007-04-23 08:17:00 +0100 | [diff] [blame] | 227 | int msi_addstringW( string_table *st, UINT n, const WCHAR *data, int len, UINT refcount, enum StringPersistence persistence ) |
Mike McCormack | 068b4ec | 2004-03-19 01:16:36 +0000 | [diff] [blame] | 228 | { |
Mike McCormack | f537eb2 | 2006-04-03 18:48:52 +0900 | [diff] [blame] | 229 | LPWSTR str; |
| 230 | |
Mike McCormack | 068b4ec | 2004-03-19 01:16:36 +0000 | [diff] [blame] | 231 | /* TRACE("[%2d] = %s\n", string_no, debugstr_an(data,len) ); */ |
| 232 | |
Mike McCormack | de8674e | 2004-06-30 18:42:02 +0000 | [diff] [blame] | 233 | if( !data ) |
| 234 | return 0; |
Mike McCormack | 068b4ec | 2004-03-19 01:16:36 +0000 | [diff] [blame] | 235 | if( !data[0] ) |
| 236 | return 0; |
| 237 | if( n > 0 ) |
| 238 | { |
Rob Shearman | 2e3289c | 2007-04-23 08:17:00 +0100 | [diff] [blame] | 239 | if( st->strings[n].persistent_refcount || |
| 240 | st->strings[n].nonpersistent_refcount ) |
Mike McCormack | 068b4ec | 2004-03-19 01:16:36 +0000 | [diff] [blame] | 241 | return -1; |
| 242 | } |
| 243 | else |
| 244 | { |
Mike McCormack | ab519f2 | 2004-06-30 18:18:27 +0000 | [diff] [blame] | 245 | if( ERROR_SUCCESS == msi_string2idW( st, data, &n ) ) |
Mike McCormack | 068b4ec | 2004-03-19 01:16:36 +0000 | [diff] [blame] | 246 | { |
Rob Shearman | 2e3289c | 2007-04-23 08:17:00 +0100 | [diff] [blame] | 247 | if (persistence == StringPersistent) |
| 248 | st->strings[n].persistent_refcount += refcount; |
| 249 | else |
| 250 | st->strings[n].nonpersistent_refcount += refcount; |
Mike McCormack | 068b4ec | 2004-03-19 01:16:36 +0000 | [diff] [blame] | 251 | return n; |
| 252 | } |
| 253 | n = st_find_free_entry( st ); |
| 254 | if( n < 0 ) |
| 255 | return -1; |
| 256 | } |
| 257 | |
Mike McCormack | abd259f | 2004-06-30 18:24:33 +0000 | [diff] [blame] | 258 | if( n < 1 ) |
| 259 | { |
| 260 | ERR("invalid index adding %s (%d)\n", debugstr_w( data ), n ); |
| 261 | return -1; |
| 262 | } |
| 263 | |
Mike McCormack | 068b4ec | 2004-03-19 01:16:36 +0000 | [diff] [blame] | 264 | /* allocate a new string */ |
Mike McCormack | ab519f2 | 2004-06-30 18:18:27 +0000 | [diff] [blame] | 265 | if(len<0) |
| 266 | len = strlenW(data); |
| 267 | TRACE("%s, n = %d len = %d\n", debugstr_w(data), n, len ); |
| 268 | |
Mike McCormack | f537eb2 | 2006-04-03 18:48:52 +0900 | [diff] [blame] | 269 | str = msi_alloc( (len+1)*sizeof(WCHAR) ); |
| 270 | if( !str ) |
Mike McCormack | 068b4ec | 2004-03-19 01:16:36 +0000 | [diff] [blame] | 271 | return -1; |
Mike McCormack | ab519f2 | 2004-06-30 18:18:27 +0000 | [diff] [blame] | 272 | TRACE("%d\n",__LINE__); |
Mike McCormack | f537eb2 | 2006-04-03 18:48:52 +0900 | [diff] [blame] | 273 | memcpy( str, data, len*sizeof(WCHAR) ); |
| 274 | str[len] = 0; |
Mike McCormack | 068b4ec | 2004-03-19 01:16:36 +0000 | [diff] [blame] | 275 | |
Rob Shearman | 2e3289c | 2007-04-23 08:17:00 +0100 | [diff] [blame] | 276 | set_st_entry( st, n, str, refcount, persistence ); |
Mike McCormack | 068b4ec | 2004-03-19 01:16:36 +0000 | [diff] [blame] | 277 | |
| 278 | return n; |
| 279 | } |
| 280 | |
Mike McCormack | 943a71e | 2004-03-19 19:14:12 +0000 | [diff] [blame] | 281 | /* find the string identified by an id - return null if there's none */ |
Andrew Talbot | 8b362f7 | 2007-06-12 21:51:36 +0100 | [diff] [blame] | 282 | const WCHAR *msi_string_lookup_id( const string_table *st, UINT id ) |
Mike McCormack | b040e4b | 2004-03-18 04:04:08 +0000 | [diff] [blame] | 283 | { |
Mike McCormack | ab519f2 | 2004-06-30 18:18:27 +0000 | [diff] [blame] | 284 | static const WCHAR zero[] = { 0 }; |
Mike McCormack | 943a71e | 2004-03-19 19:14:12 +0000 | [diff] [blame] | 285 | if( id == 0 ) |
Mike McCormack | ab519f2 | 2004-06-30 18:18:27 +0000 | [diff] [blame] | 286 | return zero; |
Mike McCormack | b040e4b | 2004-03-18 04:04:08 +0000 | [diff] [blame] | 287 | |
Mike McCormack | abd259f | 2004-06-30 18:24:33 +0000 | [diff] [blame] | 288 | if( id >= st->maxcount ) |
Mike McCormack | 943a71e | 2004-03-19 19:14:12 +0000 | [diff] [blame] | 289 | return NULL; |
Mike McCormack | b040e4b | 2004-03-18 04:04:08 +0000 | [diff] [blame] | 290 | |
Rob Shearman | 2e3289c | 2007-04-23 08:17:00 +0100 | [diff] [blame] | 291 | if( id && !st->strings[id].persistent_refcount && !st->strings[id].nonpersistent_refcount) |
Mike McCormack | 943a71e | 2004-03-19 19:14:12 +0000 | [diff] [blame] | 292 | return NULL; |
Mike McCormack | b040e4b | 2004-03-18 04:04:08 +0000 | [diff] [blame] | 293 | |
Mike McCormack | 943a71e | 2004-03-19 19:14:12 +0000 | [diff] [blame] | 294 | return st->strings[id].str; |
Mike McCormack | b040e4b | 2004-03-18 04:04:08 +0000 | [diff] [blame] | 295 | } |
| 296 | |
Mike McCormack | 943a71e | 2004-03-19 19:14:12 +0000 | [diff] [blame] | 297 | /* |
| 298 | * msi_id2stringW |
| 299 | * |
| 300 | * [in] st - pointer to the string table |
Francois Gouget | 497709b | 2004-06-15 20:26:45 +0000 | [diff] [blame] | 301 | * [in] id - id of the string to retrieve |
Mike McCormack | 943a71e | 2004-03-19 19:14:12 +0000 | [diff] [blame] | 302 | * [out] buffer - destination of the string |
| 303 | * [in/out] sz - number of bytes available in the buffer on input |
| 304 | * number of bytes used on output |
| 305 | * |
| 306 | * The size includes the terminating nul character. Short buffers |
| 307 | * will be filled, but not nul terminated. |
| 308 | */ |
Andrew Talbot | 8b362f7 | 2007-06-12 21:51:36 +0100 | [diff] [blame] | 309 | UINT msi_id2stringW( const string_table *st, UINT id, LPWSTR buffer, UINT *sz ) |
Mike McCormack | f5e1c13 | 2004-03-17 20:49:59 +0000 | [diff] [blame] | 310 | { |
| 311 | UINT len; |
Mike McCormack | ab519f2 | 2004-06-30 18:18:27 +0000 | [diff] [blame] | 312 | const WCHAR *str; |
Mike McCormack | f5e1c13 | 2004-03-17 20:49:59 +0000 | [diff] [blame] | 313 | |
Mike McCormack | abd259f | 2004-06-30 18:24:33 +0000 | [diff] [blame] | 314 | TRACE("Finding string %d of %d\n", id, st->maxcount); |
Mike McCormack | 943a71e | 2004-03-19 19:14:12 +0000 | [diff] [blame] | 315 | |
Mike McCormack | 9d66d94 | 2004-06-26 00:11:08 +0000 | [diff] [blame] | 316 | str = msi_string_lookup_id( st, id ); |
Mike McCormack | 943a71e | 2004-03-19 19:14:12 +0000 | [diff] [blame] | 317 | if( !str ) |
Mike McCormack | f5e1c13 | 2004-03-17 20:49:59 +0000 | [diff] [blame] | 318 | return ERROR_FUNCTION_FAILED; |
| 319 | |
Mike McCormack | ab519f2 | 2004-06-30 18:18:27 +0000 | [diff] [blame] | 320 | len = strlenW( str ) + 1; |
Mike McCormack | f5e1c13 | 2004-03-17 20:49:59 +0000 | [diff] [blame] | 321 | |
| 322 | if( !buffer ) |
| 323 | { |
| 324 | *sz = len; |
| 325 | return ERROR_SUCCESS; |
| 326 | } |
| 327 | |
Mike McCormack | ab519f2 | 2004-06-30 18:18:27 +0000 | [diff] [blame] | 328 | if( *sz < len ) |
| 329 | *sz = len; |
| 330 | memcpy( buffer, str, (*sz)*sizeof(WCHAR) ); |
| 331 | *sz = len; |
Mike McCormack | f5e1c13 | 2004-03-17 20:49:59 +0000 | [diff] [blame] | 332 | |
| 333 | return ERROR_SUCCESS; |
| 334 | } |
| 335 | |
Mike McCormack | 943a71e | 2004-03-19 19:14:12 +0000 | [diff] [blame] | 336 | /* |
| 337 | * msi_id2stringA |
| 338 | * |
| 339 | * [in] st - pointer to the string table |
Francois Gouget | 497709b | 2004-06-15 20:26:45 +0000 | [diff] [blame] | 340 | * [in] id - id of the string to retrieve |
Mike McCormack | 943a71e | 2004-03-19 19:14:12 +0000 | [diff] [blame] | 341 | * [out] buffer - destination of the UTF8 string |
| 342 | * [in/out] sz - number of bytes available in the buffer on input |
| 343 | * number of bytes used on output |
| 344 | * |
| 345 | * The size includes the terminating nul character. Short buffers |
| 346 | * will be filled, but not nul terminated. |
| 347 | */ |
Andrew Talbot | 8b362f7 | 2007-06-12 21:51:36 +0100 | [diff] [blame] | 348 | UINT msi_id2stringA( const string_table *st, UINT id, LPSTR buffer, UINT *sz ) |
Mike McCormack | 943a71e | 2004-03-19 19:14:12 +0000 | [diff] [blame] | 349 | { |
| 350 | UINT len; |
Mike McCormack | ab519f2 | 2004-06-30 18:18:27 +0000 | [diff] [blame] | 351 | const WCHAR *str; |
Mike McCormack | abd259f | 2004-06-30 18:24:33 +0000 | [diff] [blame] | 352 | int n; |
Mike McCormack | 943a71e | 2004-03-19 19:14:12 +0000 | [diff] [blame] | 353 | |
Mike McCormack | abd259f | 2004-06-30 18:24:33 +0000 | [diff] [blame] | 354 | TRACE("Finding string %d of %d\n", id, st->maxcount); |
Mike McCormack | 943a71e | 2004-03-19 19:14:12 +0000 | [diff] [blame] | 355 | |
Mike McCormack | 9d66d94 | 2004-06-26 00:11:08 +0000 | [diff] [blame] | 356 | str = msi_string_lookup_id( st, id ); |
Mike McCormack | 943a71e | 2004-03-19 19:14:12 +0000 | [diff] [blame] | 357 | if( !str ) |
| 358 | return ERROR_FUNCTION_FAILED; |
| 359 | |
Mike McCormack | ab519f2 | 2004-06-30 18:18:27 +0000 | [diff] [blame] | 360 | len = WideCharToMultiByte( st->codepage, 0, str, -1, NULL, 0, NULL, NULL ); |
Mike McCormack | 943a71e | 2004-03-19 19:14:12 +0000 | [diff] [blame] | 361 | |
| 362 | if( !buffer ) |
| 363 | { |
| 364 | *sz = len; |
| 365 | return ERROR_SUCCESS; |
| 366 | } |
| 367 | |
Mike McCormack | abd259f | 2004-06-30 18:24:33 +0000 | [diff] [blame] | 368 | if( len > *sz ) |
| 369 | { |
| 370 | n = strlenW( str ) + 1; |
| 371 | while( n && (len > *sz) ) |
| 372 | len = WideCharToMultiByte( st->codepage, 0, |
| 373 | str, --n, NULL, 0, NULL, NULL ); |
| 374 | } |
| 375 | else |
| 376 | n = -1; |
| 377 | |
| 378 | *sz = WideCharToMultiByte( st->codepage, 0, str, n, buffer, len, NULL, NULL ); |
Mike McCormack | 943a71e | 2004-03-19 19:14:12 +0000 | [diff] [blame] | 379 | |
| 380 | return ERROR_SUCCESS; |
| 381 | } |
| 382 | |
| 383 | /* |
Mike McCormack | ab519f2 | 2004-06-30 18:18:27 +0000 | [diff] [blame] | 384 | * msi_string2idW |
Mike McCormack | 943a71e | 2004-03-19 19:14:12 +0000 | [diff] [blame] | 385 | * |
| 386 | * [in] st - pointer to the string table |
Mike McCormack | ab519f2 | 2004-06-30 18:18:27 +0000 | [diff] [blame] | 387 | * [in] str - string to find in the string table |
Mike McCormack | 943a71e | 2004-03-19 19:14:12 +0000 | [diff] [blame] | 388 | * [out] id - id of the string, if found |
| 389 | */ |
Andrew Talbot | 8b362f7 | 2007-06-12 21:51:36 +0100 | [diff] [blame] | 390 | UINT msi_string2idW( const string_table *st, LPCWSTR str, UINT *id ) |
Mike McCormack | f5e1c13 | 2004-03-17 20:49:59 +0000 | [diff] [blame] | 391 | { |
Mike McCormack | f537eb2 | 2006-04-03 18:48:52 +0900 | [diff] [blame] | 392 | UINT n, hash = msistring_makehash( str ); |
| 393 | msistring *se = st->strings; |
Mike McCormack | f5e1c13 | 2004-03-17 20:49:59 +0000 | [diff] [blame] | 394 | |
Mike McCormack | f537eb2 | 2006-04-03 18:48:52 +0900 | [diff] [blame] | 395 | for (n = st->hash[hash]; n != -1; n = st->strings[n].hash_next ) |
Mike McCormack | f5e1c13 | 2004-03-17 20:49:59 +0000 | [diff] [blame] | 396 | { |
Mike McCormack | f537eb2 | 2006-04-03 18:48:52 +0900 | [diff] [blame] | 397 | if ((str == se[n].str) || !lstrcmpW(str, se[n].str)) |
Mike McCormack | f5e1c13 | 2004-03-17 20:49:59 +0000 | [diff] [blame] | 398 | { |
Mike McCormack | f537eb2 | 2006-04-03 18:48:52 +0900 | [diff] [blame] | 399 | *id = n; |
| 400 | return ERROR_SUCCESS; |
Mike McCormack | f5e1c13 | 2004-03-17 20:49:59 +0000 | [diff] [blame] | 401 | } |
| 402 | } |
| 403 | |
Mike McCormack | f537eb2 | 2006-04-03 18:48:52 +0900 | [diff] [blame] | 404 | return ERROR_INVALID_PARAMETER; |
Mike McCormack | 068b4ec | 2004-03-19 01:16:36 +0000 | [diff] [blame] | 405 | } |
| 406 | |
Andrew Talbot | 8b362f7 | 2007-06-12 21:51:36 +0100 | [diff] [blame] | 407 | UINT msi_string2idA( const string_table *st, LPCSTR buffer, UINT *id ) |
Mike McCormack | 068b4ec | 2004-03-19 01:16:36 +0000 | [diff] [blame] | 408 | { |
| 409 | DWORD sz; |
| 410 | UINT r = ERROR_INVALID_PARAMETER; |
Mike McCormack | ab519f2 | 2004-06-30 18:18:27 +0000 | [diff] [blame] | 411 | LPWSTR str; |
Mike McCormack | 068b4ec | 2004-03-19 01:16:36 +0000 | [diff] [blame] | 412 | |
Mike McCormack | ab519f2 | 2004-06-30 18:18:27 +0000 | [diff] [blame] | 413 | TRACE("Finding string %s in string table\n", debugstr_a(buffer) ); |
Mike McCormack | 068b4ec | 2004-03-19 01:16:36 +0000 | [diff] [blame] | 414 | |
Mike McCormack | 943a71e | 2004-03-19 19:14:12 +0000 | [diff] [blame] | 415 | if( buffer[0] == 0 ) |
| 416 | { |
| 417 | *id = 0; |
| 418 | return ERROR_SUCCESS; |
| 419 | } |
| 420 | |
Mike McCormack | ab519f2 | 2004-06-30 18:18:27 +0000 | [diff] [blame] | 421 | sz = MultiByteToWideChar( st->codepage, 0, buffer, -1, NULL, 0 ); |
Mike McCormack | 068b4ec | 2004-03-19 01:16:36 +0000 | [diff] [blame] | 422 | if( sz <= 0 ) |
| 423 | return r; |
Mike McCormack | 8dc28d5 | 2005-09-20 11:57:19 +0000 | [diff] [blame] | 424 | str = msi_alloc( sz*sizeof(WCHAR) ); |
Mike McCormack | 068b4ec | 2004-03-19 01:16:36 +0000 | [diff] [blame] | 425 | if( !str ) |
| 426 | return ERROR_NOT_ENOUGH_MEMORY; |
Mike McCormack | ab519f2 | 2004-06-30 18:18:27 +0000 | [diff] [blame] | 427 | MultiByteToWideChar( st->codepage, 0, buffer, -1, str, sz ); |
Mike McCormack | 068b4ec | 2004-03-19 01:16:36 +0000 | [diff] [blame] | 428 | |
Mike McCormack | ab519f2 | 2004-06-30 18:18:27 +0000 | [diff] [blame] | 429 | r = msi_string2idW( st, str, id ); |
Mike McCormack | 8dc28d5 | 2005-09-20 11:57:19 +0000 | [diff] [blame] | 430 | msi_free( str ); |
Mike McCormack | f5e1c13 | 2004-03-17 20:49:59 +0000 | [diff] [blame] | 431 | |
| 432 | return r; |
| 433 | } |
Mike McCormack | b040e4b | 2004-03-18 04:04:08 +0000 | [diff] [blame] | 434 | |
Andrew Talbot | 8b362f7 | 2007-06-12 21:51:36 +0100 | [diff] [blame] | 435 | UINT msi_strcmp( const string_table *st, UINT lval, UINT rval, UINT *res ) |
Mike McCormack | 9d66d94 | 2004-06-26 00:11:08 +0000 | [diff] [blame] | 436 | { |
Mike McCormack | ab519f2 | 2004-06-30 18:18:27 +0000 | [diff] [blame] | 437 | const WCHAR *l_str, *r_str; |
Mike McCormack | 9d66d94 | 2004-06-26 00:11:08 +0000 | [diff] [blame] | 438 | |
| 439 | l_str = msi_string_lookup_id( st, lval ); |
| 440 | if( !l_str ) |
| 441 | return ERROR_INVALID_PARAMETER; |
| 442 | |
| 443 | r_str = msi_string_lookup_id( st, rval ); |
| 444 | if( !r_str ) |
| 445 | return ERROR_INVALID_PARAMETER; |
| 446 | |
| 447 | /* does this do the right thing for all UTF-8 strings? */ |
Mike McCormack | ab519f2 | 2004-06-30 18:18:27 +0000 | [diff] [blame] | 448 | *res = strcmpW( l_str, r_str ); |
Mike McCormack | 9d66d94 | 2004-06-26 00:11:08 +0000 | [diff] [blame] | 449 | |
| 450 | return ERROR_SUCCESS; |
| 451 | } |
Mike McCormack | 068b4ec | 2004-03-19 01:16:36 +0000 | [diff] [blame] | 452 | |
Andrew Talbot | 8b362f7 | 2007-06-12 21:51:36 +0100 | [diff] [blame] | 453 | static void string_totalsize( const string_table *st, UINT *datasize, UINT *poolsize ) |
Mike McCormack | b040e4b | 2004-03-18 04:04:08 +0000 | [diff] [blame] | 454 | { |
Mike McCormack | 5f832b2 | 2006-08-28 16:51:41 +0900 | [diff] [blame] | 455 | UINT i, len, max, holesize; |
Mike McCormack | b040e4b | 2004-03-18 04:04:08 +0000 | [diff] [blame] | 456 | |
Rob Shearman | 2e3289c | 2007-04-23 08:17:00 +0100 | [diff] [blame] | 457 | if( st->strings[0].str || st->strings[0].persistent_refcount || st->strings[0].nonpersistent_refcount) |
Mike McCormack | abd259f | 2004-06-30 18:24:33 +0000 | [diff] [blame] | 458 | ERR("oops. element 0 has a string\n"); |
Mike McCormack | 5f832b2 | 2006-08-28 16:51:41 +0900 | [diff] [blame] | 459 | |
| 460 | *poolsize = 4; |
| 461 | *datasize = 0; |
| 462 | max = 1; |
| 463 | holesize = 0; |
Mike McCormack | abd259f | 2004-06-30 18:24:33 +0000 | [diff] [blame] | 464 | for( i=1; i<st->maxcount; i++ ) |
Mike McCormack | b040e4b | 2004-03-18 04:04:08 +0000 | [diff] [blame] | 465 | { |
Rob Shearman | 2e3289c | 2007-04-23 08:17:00 +0100 | [diff] [blame] | 466 | if( !st->strings[i].persistent_refcount ) |
| 467 | continue; |
Mike McCormack | b040e4b | 2004-03-18 04:04:08 +0000 | [diff] [blame] | 468 | if( st->strings[i].str ) |
Mike McCormack | ab519f2 | 2004-06-30 18:18:27 +0000 | [diff] [blame] | 469 | { |
Mike McCormack | abd259f | 2004-06-30 18:24:33 +0000 | [diff] [blame] | 470 | TRACE("[%u] = %s\n", i, debugstr_w(st->strings[i].str)); |
Mike McCormack | ab519f2 | 2004-06-30 18:18:27 +0000 | [diff] [blame] | 471 | len = WideCharToMultiByte( st->codepage, 0, |
| 472 | st->strings[i].str, -1, NULL, 0, NULL, NULL); |
| 473 | if( len ) |
| 474 | len--; |
Mike McCormack | 5f832b2 | 2006-08-28 16:51:41 +0900 | [diff] [blame] | 475 | (*datasize) += len; |
| 476 | if (len>0xffff) |
| 477 | (*poolsize) += 4; |
| 478 | max = i + 1; |
| 479 | (*poolsize) += holesize + 4; |
| 480 | holesize = 0; |
Mike McCormack | ab519f2 | 2004-06-30 18:18:27 +0000 | [diff] [blame] | 481 | } |
Mike McCormack | 5f832b2 | 2006-08-28 16:51:41 +0900 | [diff] [blame] | 482 | else |
| 483 | holesize += 4; |
Mike McCormack | b040e4b | 2004-03-18 04:04:08 +0000 | [diff] [blame] | 484 | } |
Mike McCormack | 5f832b2 | 2006-08-28 16:51:41 +0900 | [diff] [blame] | 485 | TRACE("data %u pool %u codepage %x\n", *datasize, *poolsize, st->codepage ); |
Mike McCormack | abd259f | 2004-06-30 18:24:33 +0000 | [diff] [blame] | 486 | } |
Rob Shearman | 37b11ba | 2007-04-23 08:17:40 +0100 | [diff] [blame] | 487 | |
| 488 | static const WCHAR szStringData[] = { |
| 489 | '_','S','t','r','i','n','g','D','a','t','a',0 }; |
| 490 | static const WCHAR szStringPool[] = { |
| 491 | '_','S','t','r','i','n','g','P','o','o','l',0 }; |
| 492 | |
| 493 | HRESULT msi_init_string_table( IStorage *stg ) |
| 494 | { |
| 495 | USHORT zero[2] = { 0, 0 }; |
| 496 | UINT ret; |
| 497 | |
| 498 | /* create the StringPool stream... add the zero string to it*/ |
James Hawkins | da55285 | 2007-04-25 05:22:01 -0500 | [diff] [blame] | 499 | ret = write_stream_data(stg, szStringPool, zero, sizeof zero, TRUE); |
Rob Shearman | 37b11ba | 2007-04-23 08:17:40 +0100 | [diff] [blame] | 500 | if (ret != ERROR_SUCCESS) |
| 501 | return E_FAIL; |
| 502 | |
| 503 | /* create the StringData stream... make it zero length */ |
James Hawkins | da55285 | 2007-04-25 05:22:01 -0500 | [diff] [blame] | 504 | ret = write_stream_data(stg, szStringData, NULL, 0, TRUE); |
Rob Shearman | 37b11ba | 2007-04-23 08:17:40 +0100 | [diff] [blame] | 505 | if (ret != ERROR_SUCCESS) |
| 506 | return E_FAIL; |
| 507 | |
| 508 | return S_OK; |
| 509 | } |
| 510 | |
James Hawkins | a05613a | 2007-06-07 16:41:51 -0700 | [diff] [blame] | 511 | string_table *msi_load_string_table( IStorage *stg, UINT *bytes_per_strref ) |
Rob Shearman | 37b11ba | 2007-04-23 08:17:40 +0100 | [diff] [blame] | 512 | { |
| 513 | string_table *st = NULL; |
| 514 | CHAR *data = NULL; |
| 515 | USHORT *pool = NULL; |
| 516 | UINT r, datasize = 0, poolsize = 0, codepage; |
| 517 | DWORD i, count, offset, len, n, refs; |
| 518 | |
James Hawkins | a05613a | 2007-06-07 16:41:51 -0700 | [diff] [blame] | 519 | static const USHORT large_str_sig[] = { 0x0000, 0x8000 }; |
| 520 | |
Rob Shearman | 37b11ba | 2007-04-23 08:17:40 +0100 | [diff] [blame] | 521 | r = read_stream_data( stg, szStringPool, &pool, &poolsize ); |
| 522 | if( r != ERROR_SUCCESS) |
| 523 | goto end; |
| 524 | r = read_stream_data( stg, szStringData, (USHORT**)&data, &datasize ); |
| 525 | if( r != ERROR_SUCCESS) |
| 526 | goto end; |
| 527 | |
James Hawkins | a05613a | 2007-06-07 16:41:51 -0700 | [diff] [blame] | 528 | if ( !memcmp(pool, large_str_sig, sizeof(large_str_sig)) ) |
| 529 | *bytes_per_strref = LONG_STR_BYTES; |
| 530 | else |
| 531 | *bytes_per_strref = sizeof(USHORT); |
| 532 | |
| 533 | /* FIXME: don't know where the codepage is in large str tables */ |
Rob Shearman | 37b11ba | 2007-04-23 08:17:40 +0100 | [diff] [blame] | 534 | count = poolsize/4; |
James Hawkins | a05613a | 2007-06-07 16:41:51 -0700 | [diff] [blame] | 535 | if( poolsize > 4 && *bytes_per_strref != LONG_STR_BYTES ) |
Rob Shearman | 37b11ba | 2007-04-23 08:17:40 +0100 | [diff] [blame] | 536 | codepage = pool[0] | ( pool[1] << 16 ); |
| 537 | else |
| 538 | codepage = CP_ACP; |
Rob Shearman | e261d60 | 2007-04-23 08:18:27 +0100 | [diff] [blame] | 539 | st = init_stringtable( count, codepage ); |
Rob Shearman | 37b11ba | 2007-04-23 08:17:40 +0100 | [diff] [blame] | 540 | |
| 541 | offset = 0; |
| 542 | n = 1; |
| 543 | i = 1; |
| 544 | while( i<count ) |
| 545 | { |
| 546 | /* the string reference count is always the second word */ |
| 547 | refs = pool[i*2+1]; |
| 548 | |
| 549 | /* empty entries have two zeros, still have a string id */ |
| 550 | if (pool[i*2] == 0 && refs == 0) |
| 551 | { |
| 552 | i++; |
| 553 | n++; |
| 554 | continue; |
| 555 | } |
| 556 | |
| 557 | /* |
| 558 | * If a string is over 64k, the previous string entry is made null |
| 559 | * and its the high word of the length is inserted in the null string's |
| 560 | * reference count field. |
| 561 | */ |
| 562 | if( pool[i*2] == 0) |
| 563 | { |
| 564 | len = (pool[i*2+3] << 16) + pool[i*2+2]; |
| 565 | i += 2; |
| 566 | } |
| 567 | else |
| 568 | { |
| 569 | len = pool[i*2]; |
| 570 | i += 1; |
| 571 | } |
| 572 | |
| 573 | if ( (offset + len) > datasize ) |
| 574 | { |
| 575 | ERR("string table corrupt?\n"); |
| 576 | break; |
| 577 | } |
| 578 | |
| 579 | r = msi_addstring( st, n, data+offset, len, refs, StringPersistent ); |
| 580 | if( r != n ) |
| 581 | ERR("Failed to add string %d\n", n ); |
| 582 | n++; |
| 583 | offset += len; |
| 584 | } |
| 585 | |
| 586 | if ( datasize != offset ) |
| 587 | ERR("string table load failed! (%08x != %08x), please report\n", datasize, offset ); |
| 588 | |
| 589 | TRACE("Loaded %d strings\n", count); |
| 590 | |
| 591 | end: |
| 592 | msi_free( pool ); |
| 593 | msi_free( data ); |
| 594 | |
| 595 | return st; |
| 596 | } |
| 597 | |
Andrew Talbot | 8b362f7 | 2007-06-12 21:51:36 +0100 | [diff] [blame] | 598 | UINT msi_save_string_table( const string_table *st, IStorage *storage ) |
Rob Shearman | 37b11ba | 2007-04-23 08:17:40 +0100 | [diff] [blame] | 599 | { |
Rob Shearman | e261d60 | 2007-04-23 08:18:27 +0100 | [diff] [blame] | 600 | UINT i, datasize = 0, poolsize = 0, sz, used, r, codepage, n; |
Rob Shearman | 37b11ba | 2007-04-23 08:17:40 +0100 | [diff] [blame] | 601 | UINT ret = ERROR_FUNCTION_FAILED; |
| 602 | CHAR *data = NULL; |
| 603 | USHORT *pool = NULL; |
| 604 | |
| 605 | TRACE("\n"); |
| 606 | |
| 607 | /* construct the new table in memory first */ |
Rob Shearman | e261d60 | 2007-04-23 08:18:27 +0100 | [diff] [blame] | 608 | string_totalsize( st, &datasize, &poolsize ); |
Rob Shearman | 37b11ba | 2007-04-23 08:17:40 +0100 | [diff] [blame] | 609 | |
Rob Shearman | e261d60 | 2007-04-23 08:18:27 +0100 | [diff] [blame] | 610 | TRACE("%u %u %u\n", st->maxcount, datasize, poolsize ); |
Rob Shearman | 37b11ba | 2007-04-23 08:17:40 +0100 | [diff] [blame] | 611 | |
| 612 | pool = msi_alloc( poolsize ); |
| 613 | if( ! pool ) |
| 614 | { |
| 615 | WARN("Failed to alloc pool %d bytes\n", poolsize ); |
| 616 | goto err; |
| 617 | } |
| 618 | data = msi_alloc( datasize ); |
| 619 | if( ! data ) |
| 620 | { |
| 621 | WARN("Failed to alloc data %d bytes\n", poolsize ); |
| 622 | goto err; |
| 623 | } |
| 624 | |
| 625 | used = 0; |
Rob Shearman | e261d60 | 2007-04-23 08:18:27 +0100 | [diff] [blame] | 626 | codepage = st->codepage; |
Rob Shearman | 37b11ba | 2007-04-23 08:17:40 +0100 | [diff] [blame] | 627 | pool[0]=codepage&0xffff; |
| 628 | pool[1]=(codepage>>16); |
| 629 | n = 1; |
Rob Shearman | e261d60 | 2007-04-23 08:18:27 +0100 | [diff] [blame] | 630 | for( i=1; i<st->maxcount; i++ ) |
Rob Shearman | 37b11ba | 2007-04-23 08:17:40 +0100 | [diff] [blame] | 631 | { |
Rob Shearman | e261d60 | 2007-04-23 08:18:27 +0100 | [diff] [blame] | 632 | if( !st->strings[i].persistent_refcount ) |
Rob Shearman | 37b11ba | 2007-04-23 08:17:40 +0100 | [diff] [blame] | 633 | continue; |
| 634 | sz = datasize - used; |
| 635 | r = msi_id2stringA( st, i, data+used, &sz ); |
| 636 | if( r != ERROR_SUCCESS ) |
| 637 | { |
| 638 | ERR("failed to fetch string\n"); |
| 639 | sz = 0; |
| 640 | } |
| 641 | if( sz && (sz < (datasize - used ) ) ) |
| 642 | sz--; |
| 643 | |
| 644 | if (sz) |
Rob Shearman | e261d60 | 2007-04-23 08:18:27 +0100 | [diff] [blame] | 645 | pool[ n*2 + 1 ] = st->strings[i].persistent_refcount; |
Rob Shearman | 37b11ba | 2007-04-23 08:17:40 +0100 | [diff] [blame] | 646 | else |
| 647 | pool[ n*2 + 1 ] = 0; |
| 648 | if (sz < 0x10000) |
| 649 | { |
| 650 | pool[ n*2 ] = sz; |
| 651 | n++; |
| 652 | } |
| 653 | else |
| 654 | { |
| 655 | pool[ n*2 ] = 0; |
| 656 | pool[ n*2 + 2 ] = sz&0xffff; |
| 657 | pool[ n*2 + 3 ] = (sz>>16); |
| 658 | n += 2; |
| 659 | } |
| 660 | used += sz; |
| 661 | if( used > datasize ) |
| 662 | { |
| 663 | ERR("oops overran %d >= %d\n", used, datasize); |
| 664 | goto err; |
| 665 | } |
| 666 | } |
| 667 | |
| 668 | if( used != datasize ) |
| 669 | { |
| 670 | ERR("oops used %d != datasize %d\n", used, datasize); |
| 671 | goto err; |
| 672 | } |
| 673 | |
| 674 | /* write the streams */ |
James Hawkins | da55285 | 2007-04-25 05:22:01 -0500 | [diff] [blame] | 675 | r = write_stream_data( storage, szStringData, data, datasize, TRUE ); |
Rob Shearman | 37b11ba | 2007-04-23 08:17:40 +0100 | [diff] [blame] | 676 | TRACE("Wrote StringData r=%08x\n", r); |
| 677 | if( r ) |
| 678 | goto err; |
James Hawkins | da55285 | 2007-04-25 05:22:01 -0500 | [diff] [blame] | 679 | r = write_stream_data( storage, szStringPool, pool, poolsize, TRUE ); |
Rob Shearman | 37b11ba | 2007-04-23 08:17:40 +0100 | [diff] [blame] | 680 | TRACE("Wrote StringPool r=%08x\n", r); |
| 681 | if( r ) |
| 682 | goto err; |
| 683 | |
| 684 | ret = ERROR_SUCCESS; |
| 685 | |
| 686 | err: |
| 687 | msi_free( data ); |
| 688 | msi_free( pool ); |
| 689 | |
| 690 | return ret; |
| 691 | } |