msvcp90: Add an msvcp-specific size_t to avoid casts in traces.
[wine] / dlls / msvcp90 / string.c
1 /*
2  * Copyright 2010 Piotr Caban for CodeWeavers
3  *
4  * This library is free software; you can redistribute it and/or
5  * modify it under the terms of the GNU Lesser General Public
6  * License as published by the Free Software Foundation; either
7  * version 2.1 of the License, or (at your option) any later version.
8  *
9  * This library is distributed in the hope that it will be useful,
10  * but WITHOUT ANY WARRANTY; without even the implied warranty of
11  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
12  * Lesser General Public License for more details.
13  *
14  * You should have received a copy of the GNU Lesser General Public
15  * License along with this library; if not, write to the Free Software
16  * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
17  */
18
19 #include "config.h"
20
21 #include <stdarg.h>
22
23 #include "msvcp90.h"
24 #include "stdio.h"
25
26 #include "windef.h"
27 #include "winbase.h"
28 #include "wine/debug.h"
29 WINE_DEFAULT_DEBUG_CHANNEL(msvcp90);
30
31 /* char_traits<char> */
32 /* ?assign@?$char_traits@D@std@@SAXAADABD@Z */
33 /* ?assign@?$char_traits@D@std@@SAXAEADAEBD@Z */
34 void CDECL MSVCP_char_traits_char_assign(char *ch, const char *assign)
35 {
36     *ch = *assign;
37 }
38
39 /* ?eq@?$char_traits@D@std@@SA_NABD0@Z */
40 /* ?eq@?$char_traits@D@std@@SA_NAEBD0@Z */
41 MSVCP_bool CDECL MSVCP_char_traits_char_eq(const char *ch1, const char *ch2)
42 {
43     return *ch1 == *ch2;
44 }
45
46 /* ?lt@?$char_traits@D@std@@SA_NABD0@Z */
47 /* ?lt@?$char_traits@D@std@@SA_NAEBD0@Z */
48 MSVCP_bool CDECL MSVCP_char_traits_lt(const char *ch1, const char *ch2)
49 {
50     return *ch1 < *ch2;
51 }
52
53 /* ?compare@?$char_traits@D@std@@SAHPBD0I@Z */
54 /* ?compare@?$char_traits@D@std@@SAHPEBD0_K@Z */
55 int CDECL MSVCP_char_traits_char_compare(
56         const char *s1, const char *s2, MSVCP_size_t count)
57 {
58     int ret = memcmp(s1, s2, count);
59     return (ret>0 ? 1 : (ret<0 ? -1 : 0));
60 }
61
62 /* ?length@?$char_traits@D@std@@SAIPBD@Z */
63 /* ?length@?$char_traits@D@std@@SA_KPEBD@Z */
64 MSVCP_size_t CDECL MSVCP_char_traits_char_length(const char *str)
65 {
66     return strlen(str);
67 }
68
69 /* ?_Copy_s@?$char_traits@D@std@@SAPADPADIPBDI@Z */
70 /* ?_Copy_s@?$char_traits@D@std@@SAPEADPEAD_KPEBD1@Z */
71 char* CDECL MSVCP_char_traits_char__Copy_s(char *dest,
72         MSVCP_size_t size, const char *src, MSVCP_size_t count)
73 {
74     if(!dest || !src || size<count) {
75         if(dest && size)
76             dest[0] = '\0';
77         _invalid_parameter(NULL, NULL, NULL, 0, 0);
78         return dest;
79     }
80
81     return memcpy(dest, src, count);
82 }
83
84 /* ?copy@?$char_traits@D@std@@SAPADPADPBDI@Z */
85 /* ?copy@?$char_traits@D@std@@SAPEADPEADPEBD_K@Z */
86 char* CDECL MSVCP_char_traits_char_copy(
87         char *dest, const char *src, MSVCP_size_t count)
88 {
89     return MSVCP_char_traits_char__Copy_s(dest, count, src, count);
90 }
91
92 /* ?find@?$char_traits@D@std@@SAPBDPBDIABD@Z */
93 /* ?find@?$char_traits@D@std@@SAPEBDPEBD_KAEBD@Z */
94 const char * CDECL MSVCP_char_traits_char_find(
95         const char *str, MSVCP_size_t range, const char *c)
96 {
97     return memchr(str, *c, range);
98 }
99
100 /* ?_Move_s@?$char_traits@D@std@@SAPADPADIPBDI@Z */
101 /* ?_Move_s@?$char_traits@D@std@@SAPEADPEAD_KPEBD1@Z */
102 char* CDECL MSVCP_char_traits_char__Move_s(char *dest,
103         MSVCP_size_t size, const char *src, MSVCP_size_t count)
104 {
105     if(!dest || !src || size<count) {
106         if(dest && size)
107             dest[0] = '\0';
108         _invalid_parameter(NULL, NULL, NULL, 0, 0);
109         return dest;
110     }
111
112     return memmove(dest, src, count);
113 }
114
115 /* ?move@?$char_traits@D@std@@SAPADPADPBDI@Z */
116 /* ?move@?$char_traits@D@std@@SAPEADPEADPEBD_K@Z */
117 char* CDECL MSVCP_char_traits_char_move(
118         char *dest, const char *src, MSVCP_size_t count)
119 {
120     return MSVCP_char_traits_char__Move_s(dest, count, src, count);
121 }
122
123 /* ?assign@?$char_traits@D@std@@SAPADPADID@Z */
124 /* ?assign@?$char_traits@D@std@@SAPEADPEAD_KD@Z */
125 char* CDECL MSVCP_char_traits_char_assignn(char *str, MSVCP_size_t num, char c)
126 {
127     return memset(str, c, num);
128 }
129
130 /* ?to_char_type@?$char_traits@D@std@@SADABH@Z */
131 /* ?to_char_type@?$char_traits@D@std@@SADAEBH@Z */
132 char CDECL MSVCP_char_traits_char_to_char_type(const int *i)
133 {
134     return (char)*i;
135 }
136
137 /* ?to_int_type@?$char_traits@D@std@@SAHABD@Z */
138 /* ?to_int_type@?$char_traits@D@std@@SAHAEBD@Z */
139 int CDECL MSVCP_char_traits_char_to_int_type(const char *ch)
140 {
141     return (int)*ch;
142 }
143
144 /* ?eq_int_type@?$char_traits@D@std@@SA_NABH0@Z */
145 /* ?eq_int_type@?$char_traits@D@std@@SA_NAEBH0@Z */
146 MSVCP_bool CDECL MSVCP_char_traits_char_eq_int_type(const int *i1, const int *i2)
147 {
148     return *i1 == *i2;
149 }
150
151 /* ?eof@?$char_traits@D@std@@SAHXZ */
152 int CDECL MSVCP_char_traits_char_eof(void)
153 {
154     return EOF;
155 }
156
157 /* ?not_eof@?$char_traits@D@std@@SAHABH@Z */
158 /* ?not_eof@?$char_traits@D@std@@SAHAEBH@Z */
159 int CDECL MSVCP_char_traits_char_not_eof(int *in)
160 {
161     return (*in==EOF ? !EOF : *in);
162 }
163
164
165 /* char_traits<wchar_t> */
166 /* ?assign@?$char_traits@_W@std@@SAXAA_WAB_W@Z */
167 /* ?assign@?$char_traits@_W@std@@SAXAEA_WAEB_W@Z */
168 void CDECL MSVCP_char_traits_wchar_assign(wchar_t *ch,
169         const wchar_t *assign)
170 {
171     *ch = *assign;
172 }
173
174 /* ?eq@?$char_traits@_W@std@@SA_NAB_W0@Z */
175 /* ?eq@?$char_traits@_W@std@@SA_NAEB_W0@Z */
176 MSVCP_bool CDECL MSVCP_char_traits_wchar_eq(wchar_t *ch1, wchar_t *ch2)
177 {
178     return *ch1 == *ch2;
179 }
180
181 /* ?lt@?$char_traits@_W@std@@SA_NAB_W0@Z */
182 /* ?lt@?$char_traits@_W@std@@SA_NAEB_W0@Z */
183 MSVCP_bool CDECL MSVCP_char_traits_wchar_lt(const wchar_t *ch1,
184         const wchar_t *ch2)
185 {
186     return *ch1 < *ch2;
187 }
188
189 /* ?compare@?$char_traits@_W@std@@SAHPB_W0I@Z */
190 /* ?compare@?$char_traits@_W@std@@SAHPEB_W0_K@Z */
191 int CDECL MSVCP_char_traits_wchar_compare(const wchar_t *s1,
192         const wchar_t *s2, MSVCP_size_t count)
193 {
194     int ret = memcmp(s1, s2, sizeof(wchar_t[count]));
195     return (ret>0 ? 1 : (ret<0 ? -1 : 0));
196 }
197
198 /* ?length@?$char_traits@_W@std@@SAIPB_W@Z */
199 /* ?length@?$char_traits@_W@std@@SA_KPEB_W@Z */
200 MSVCP_size_t CDECL MSVCP_char_traits_wchar_length(const wchar_t *str)
201 {
202     return wcslen((WCHAR*)str);
203 }
204
205 /* ?_Copy_s@?$char_traits@_W@std@@SAPA_WPA_WIPB_WI@Z */
206 /* ?_Copy_s@?$char_traits@_W@std@@SAPEA_WPEA_W_KPEB_W1@Z */
207 wchar_t* CDECL MSVCP_char_traits_wchar__Copy_s(wchar_t *dest,
208         MSVCP_size_t size, const wchar_t *src, MSVCP_size_t count)
209 {
210     if(!dest || !src || size<count) {
211         if(dest && size)
212             dest[0] = '\0';
213         _invalid_parameter(NULL, NULL, NULL, 0, 0);
214         return dest;
215     }
216
217     return memcpy(dest, src, sizeof(wchar_t[count]));
218 }
219
220 /* ?copy@?$char_traits@_W@std@@SAPA_WPA_WPB_WI@Z */
221 /* ?copy@?$char_traits@_W@std@@SAPEA_WPEA_WPEB_W_K@Z */
222 wchar_t* CDECL MSVCP_char_traits_wchar_copy(wchar_t *dest,
223         const wchar_t *src, MSVCP_size_t count)
224 {
225     return MSVCP_char_traits_wchar__Copy_s(dest, count, src, count);
226 }
227
228 /* ?find@?$char_traits@_W@std@@SAPB_WPB_WIAB_W@Z */
229 /* ?find@?$char_traits@_W@std@@SAPEB_WPEB_W_KAEB_W@Z */
230 const wchar_t* CDECL MSVCP_char_traits_wchar_find(
231         const wchar_t *str, MSVCP_size_t range, const wchar_t *c)
232 {
233     MSVCP_size_t i=0;
234
235     for(i=0; i<range; i++)
236         if(str[i] == *c)
237             return str+i;
238
239     return NULL;
240 }
241
242 /* ?_Move_s@?$char_traits@_W@std@@SAPA_WPA_WIPB_WI@Z */
243 /* ?_Move_s@?$char_traits@_W@std@@SAPEA_WPEA_W_KPEB_W1@Z */
244 wchar_t* CDECL MSVCP_char_traits_wchar__Move_s(wchar_t *dest,
245         MSVCP_size_t size, const wchar_t *src, MSVCP_size_t count)
246 {
247     if(!dest || !src || size<count) {
248         if(dest && size)
249             dest[0] = '\0';
250         _invalid_parameter(NULL, NULL, NULL, 0, 0);
251         return dest;
252     }
253
254     return memmove(dest, src, sizeof(WCHAR[count]));
255 }
256
257 /* ?move@?$char_traits@_W@std@@SAPA_WPA_WPB_WI@Z */
258 /* ?move@?$char_traits@_W@std@@SAPEA_WPEA_WPEB_W_K@Z */
259 wchar_t* CDECL MSVCP_char_traits_wchar_move(wchar_t *dest,
260         const wchar_t *src, MSVCP_size_t count)
261 {
262     return MSVCP_char_traits_wchar__Move_s(dest, count, src, count);
263 }
264
265 /* ?assign@?$char_traits@_W@std@@SAPA_WPA_WI_W@Z */
266 /* ?assign@?$char_traits@_W@std@@SAPEA_WPEA_W_K_W@Z */
267 wchar_t* CDECL MSVCP_char_traits_wchar_assignn(wchar_t *str,
268         MSVCP_size_t num, wchar_t c)
269 {
270     MSVCP_size_t i;
271
272     for(i=0; i<num; i++)
273         str[i] = c;
274
275     return str;
276 }
277
278 /* ?to_char_type@?$char_traits@_W@std@@SA_WABG@Z */
279 /* ?to_char_type@?$char_traits@_W@std@@SA_WAEBG@Z */
280 wchar_t CDECL MSVCP_char_traits_wchar_to_char_type(const unsigned short *i)
281 {
282     return *i;
283 }
284
285 /* ?to_int_type@?$char_traits@_W@std@@SAGAB_W@Z */
286 /* ?to_int_type@?$char_traits@_W@std@@SAGAEB_W@Z */
287 unsigned short CDECL MSVCP_char_traits_wchar_to_int_type(const wchar_t *ch)
288 {
289     return *ch;
290 }
291
292 /* ?eq_int_type@?$char_traits@_W@std@@SA_NABG0@Z */
293 /* ?eq_int_type@?$char_traits@_W@std@@SA_NAEBG0@Z */
294 MSVCP_bool CDECL MSVCP_char_traits_wchar_eq_int_tpe(const unsigned short *i1,
295         const unsigned short *i2)
296 {
297     return *i1 == *i2;
298 }
299
300 /* ?eof@?$char_traits@_W@std@@SAGXZ */
301 unsigned short CDECL MSVCP_char_traits_wchar_eof(void)
302 {
303     return WEOF;
304 }
305
306 /* ?not_eof@?$char_traits@_W@std@@SAGABG@Z */
307 /* ?not_eof@?$char_traits@_W@std@@SAGAEBG@Z */
308 unsigned short CDECL MSVCP_char_traits_wchar_not_eof(const unsigned short *in)
309 {
310     return (*in==WEOF ? !WEOF : *in);
311 }
312
313
314 /* char_traits<unsigned short> */
315 /* ?assign@?$char_traits@G@std@@SAXAAGABG@Z */
316 /* ?assign@?$char_traits@G@std@@SAXAEAGAEBG@Z */
317 void CDECL MSVCP_char_traits_short_assign(unsigned short *ch,
318         const unsigned short *assign)
319 {
320     *ch = *assign;
321 }
322
323 /* ?eq@?$char_traits@G@std@@SA_NABG0@Z */
324 /* ?eq@?$char_traits@G@std@@SA_NAEBG0@Z */
325 MSVCP_bool CDECL MSVCP_char_traits_short_eq(const unsigned short *ch1,
326         const unsigned short *ch2)
327 {
328     return *ch1 == *ch2;
329 }
330
331 /* ?lt@?$char_traits@G@std@@SA_NABG0@Z */
332 /* ?lt@?$char_traits@G@std@@SA_NAEBG0@Z */
333 MSVCP_bool CDECL MSVCP_char_traits_short_lt(const unsigned short *ch1,
334         const unsigned short *ch2)
335 {
336     return *ch1 < *ch2;
337 }
338
339 /* ?compare@?$char_traits@G@std@@SAHPBG0I@Z */
340 /* ?compare@?$char_traits@G@std@@SAHPEBG0_K@Z */
341 int CDECL MSVCP_char_traits_short_compare(const unsigned short *s1,
342         const unsigned short *s2, MSVCP_size_t count)
343 {
344     MSVCP_size_t i;
345
346     for(i=0; i<count; i++)
347         if(s1[i] != s2[i])
348             return (s1[i] < s2[i] ? -1 : 1);
349
350     return 0;
351 }
352
353 /* ?length@?$char_traits@G@std@@SAIPBG@Z */
354 /* ?length@?$char_traits@G@std@@SA_KPEBG@Z */
355 MSVCP_size_t CDECL MSVCP_char_traits_short_length(const unsigned short *str)
356 {
357     MSVCP_size_t len;
358
359     for(len=0; str[len]; len++);
360
361     return len;
362 }
363
364 /* ?_Copy_s@?$char_traits@G@std@@SAPAGPAGIPBGI@Z */
365 /* ?_Copy_s@?$char_traits@G@std@@SAPEAGPEAG_KPEBG1@Z */
366 unsigned short * CDECL MSVCP_char_traits_short__Copy_s(unsigned short *dest,
367         MSVCP_size_t size, const unsigned short *src, MSVCP_size_t count)
368 {
369     if(size<count) {
370         _invalid_parameter(NULL, NULL, NULL, 0, 0);
371         return dest;
372     }
373
374     return memcpy(dest, src, sizeof(unsigned short[count]));
375 }
376
377 /* ?copy@?$char_traits@G@std@@SAPAGPAGPBGI@Z */
378 /* ?copy@?$char_traits@G@std@@SAPEAGPEAGPEBG_K@Z */
379 unsigned short* CDECL MSVCP_char_traits_short_copy(unsigned short *dest,
380         const unsigned short *src, MSVCP_size_t count)
381 {
382     return MSVCP_char_traits_short__Copy_s(dest, count, src, count);
383 }
384
385 /* ?find@?$char_traits@G@std@@SAPBGPBGIABG@Z */
386 /* ?find@?$char_traits@G@std@@SAPEBGPEBG_KAEBG@Z */
387 const unsigned short* CDECL MSVCP_char_traits_short_find(
388         const unsigned short *str, MSVCP_size_t range, const unsigned short *c)
389 {
390     MSVCP_size_t i;
391
392     for(i=0; i<range; i++)
393         if(str[i] == *c)
394             return str+i;
395
396     return NULL;
397 }
398
399 /* ?_Move_s@?$char_traits@G@std@@SAPAGPAGIPBGI@Z */
400 /* ?_Move_s@?$char_traits@G@std@@SAPEAGPEAG_KPEBG1@Z */
401 unsigned short* CDECL MSVCP_char_traits_short__Move_s(unsigned short *dest,
402         MSVCP_size_t size, const unsigned short *src, MSVCP_size_t count)
403 {
404     if(size<count) {
405         _invalid_parameter(NULL, NULL, NULL, 0, 0);
406         return dest;
407     }
408
409     return memmove(dest, src, sizeof(unsigned short[count]));
410 }
411
412 /* ?move@?$char_traits@G@std@@SAPAGPAGPBGI@Z */
413 /* ?move@?$char_traits@G@std@@SAPEAGPEAGPEBG_K@Z */
414 unsigned short* CDECL MSVCP_char_traits_short_move(unsigned short *dest,
415         const unsigned short *src, MSVCP_size_t count)
416 {
417     return MSVCP_char_traits_short__Move_s(dest, count, src, count);
418 }
419
420 /* ?assign@?$char_traits@G@std@@SAPAGPAGIG@Z */
421 /* ?assign@?$char_traits@G@std@@SAPEAGPEAG_KG@Z */
422 unsigned short* CDECL MSVCP_char_traits_short_assignn(unsigned short *str,
423         MSVCP_size_t num, unsigned short c)
424 {
425     MSVCP_size_t i;
426
427     for(i=0; i<num; i++)
428         str[i] = c;
429
430     return str;
431 }
432
433 /* ?to_char_type@?$char_traits@G@std@@SAGABG@Z */
434 /* ?to_char_type@?$char_traits@G@std@@SAGAEBG@Z */
435 unsigned short CDECL MSVCP_char_traits_short_to_char_type(const unsigned short *i)
436 {
437     return *i;
438 }
439
440 /* ?to_int_type@?$char_traits@G@std@@SAGABG@Z */
441 /* ?to_int_type@?$char_traits@G@std@@SAGAEBG@Z */
442 unsigned short CDECL MSVCP_char_traits_short_to_int_type(const unsigned short *ch)
443 {
444     return *ch;
445 }
446
447 /* ?eq_int_type@?$char_traits@G@std@@SA_NABG0@Z */
448 /* ?eq_int_type@?$char_traits@G@std@@SA_NAEBG0@Z */
449 MSVCP_bool CDECL MSVCP_char_traits_short_eq_int_type(unsigned short *i1,
450         unsigned short *i2)
451 {
452     return *i1 == *i2;
453 }
454
455 /* ?eof@?$char_traits@G@std@@SAGXZ */
456 unsigned short CDECL MSVCP_char_traits_short_eof(void)
457 {
458     return -1;
459 }
460
461 /* ?not_eof@?$char_traits@G@std@@SAGABG@Z */
462 /* ?not_eof@?$char_traits@G@std@@SAGAEBG@Z */
463 unsigned short CDECL MSVCP_char_traits_short_not_eof(const unsigned short *in)
464 {
465     return (*in==(unsigned short)-1 ? 0 : *in);
466 }
467
468
469 /* _String_base */
470 /* ?_Xlen@_String_base@std@@SAXXZ */
471 void  CDECL MSVCP__String_base_Xlen(void)
472 {
473     static const char msg[] = "string too long";
474
475     TRACE("\n");
476     throw_exception(EXCEPTION_LENGTH_ERROR, msg);
477 }
478
479 /* ?_Xran@_String_base@std@@SAXXZ */
480 void CDECL MSVCP__String_base_Xran(void)
481 {
482     static const char msg[] = "invalid string position";
483
484     TRACE("\n");
485     throw_exception(EXCEPTION_OUT_OF_RANGE, msg);
486 }
487
488 /* ?_Xinvarg@_String_base@std@@SAXXZ */
489 void CDECL MSVCP__String_base_Xinvarg(void)
490 {
491     static const char msg[] = "invalid string argument";
492
493     TRACE("\n");
494     throw_exception(EXCEPTION_INVALID_ARGUMENT, msg);
495 }
496
497
498 /* basic_string<char, char_traits<char>, allocator<char>> */
499 /* ?npos@?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@2IB */
500 /* ?npos@?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@2_KB */
501 const MSVCP_size_t MSVCP_basic_string_char_npos = -1;
502
503 /* Internal: basic_string_char_ptr - return pointer to stored string */
504 static char* basic_string_char_ptr(basic_string_char *this)
505 {
506     if(this->res == BUF_SIZE_CHAR-1)
507         return this->data.buf;
508     return this->data.ptr;
509 }
510
511 /* Internal: basic_string_char_const_ptr - returns const pointer to stored string */
512 static const char* basic_string_char_const_ptr(const basic_string_char *this)
513 {
514     if(this->res == BUF_SIZE_CHAR-1)
515         return this->data.buf;
516     return this->data.ptr;
517 }
518
519 /* Internal: basic_string_char_eos - sets string length, puts '\0' on the end */
520 static void basic_string_char_eos(basic_string_char *this, MSVCP_size_t len)
521 {
522     static const char nullbyte = '\0';
523
524     this->size = len;
525     MSVCP_char_traits_char_assign(basic_string_char_ptr(this)+len, &nullbyte);
526 }
527
528 /* Internal: basic_string_char_inside - checks if given pointer points inside stored string */
529 static MSVCP_bool basic_string_char_inside(
530         basic_string_char *this, const char *ptr)
531 {
532     char *cstr = basic_string_char_ptr(this);
533
534     return (ptr<cstr || ptr>=cstr+this->size) ? FALSE : TRUE;
535 }
536
537 /* Internal: basic_string_char_tidy - initialize basic_string buffer, deallocates data */
538 /* Caution: new_size have to be smaller than BUF_SIZE_CHAR */
539 static void basic_string_char_tidy(basic_string_char *this,
540         MSVCP_bool built, MSVCP_size_t new_size)
541 {
542     if(built && BUF_SIZE_CHAR<=this->res) {
543         char *ptr = this->data.ptr;
544
545         if(new_size > 0)
546             MSVCP_char_traits_char__Copy_s(this->data.buf, BUF_SIZE_CHAR, ptr, new_size);
547         MSVCP_allocator_char_deallocate(this->allocator, ptr, this->res+1);
548     }
549
550     this->res = BUF_SIZE_CHAR-1;
551     basic_string_char_eos(this, new_size);
552 }
553
554 /* Internal: basic_string_char_grow - changes size of internal buffer */
555 static MSVCP_bool basic_string_char_grow(
556         basic_string_char *this, MSVCP_size_t new_size, MSVCP_bool trim)
557 {
558     if(this->res < new_size) {
559         MSVCP_size_t new_res = new_size;
560         char *ptr;
561
562         new_res |= 0xf;
563
564         if(new_res/3 < this->res/2)
565             new_res = this->res + this->res/2;
566
567         ptr = MSVCP_allocator_char_allocate(this->allocator, new_res);
568         if(!ptr)
569             ptr = MSVCP_allocator_char_allocate(this->allocator, new_size+1);
570         else
571             new_size = new_res;
572         if(!ptr) {
573             ERR("Out of memory\n");
574             basic_string_char_tidy(this, TRUE, 0);
575             return FALSE;
576         }
577
578         MSVCP_char_traits_char__Copy_s(ptr, new_size,
579                 basic_string_char_ptr(this), this->size);
580         basic_string_char_tidy(this, TRUE, 0);
581         this->data.ptr = ptr;
582         this->res = new_size;
583         basic_string_char_eos(this, this->size);
584     } else if(trim && new_size < BUF_SIZE_CHAR)
585         basic_string_char_tidy(this, TRUE,
586                 new_size<this->size ? new_size : this->size);
587     else if(new_size == 0)
588         basic_string_char_eos(this, 0);
589
590     return (new_size>0);
591 }
592
593 /* ?erase@?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QAEAAV12@II@Z */
594 /* ?erase@?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QEAAAEAV12@_K0@Z */
595 DEFINE_THISCALL_WRAPPER(MSVCP_basic_string_char_erase, 12)
596 basic_string_char* __thiscall MSVCP_basic_string_char_erase(
597         basic_string_char *this, MSVCP_size_t pos, MSVCP_size_t len)
598 {
599     TRACE("%p %lu %lu\n", this, pos, len);
600
601     if(pos > this->size) {
602         MSVCP__String_base_Xran();
603         return NULL;
604     }
605
606     if(len > this->size-pos)
607         len = this->size-pos;
608
609     if(len) {
610         MSVCP_char_traits_char__Move_s(basic_string_char_ptr(this)+pos,
611                 this->res-pos, basic_string_char_ptr(this)+pos+len,
612                 this->size-pos-len);
613         basic_string_char_eos(this, this->size-len);
614     }
615
616     return this;
617 }
618
619 /* ?assign@?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QAEAAV12@ABV12@II@Z */
620 /* ?assign@?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QEAAAEAV12@AEBV12@_K1@Z */
621 DEFINE_THISCALL_WRAPPER(MSVCP_basic_string_char_assign_substr, 16)
622 basic_string_char* __thiscall MSVCP_basic_string_char_assign_substr(
623         basic_string_char *this, const basic_string_char *assign,
624         MSVCP_size_t pos, MSVCP_size_t len)
625 {
626     TRACE("%p %p %lu %lu\n", this, assign, pos, len);
627
628     if(assign->size < pos) {
629         MSVCP__String_base_Xran();
630         return NULL;
631     }
632
633     if(len > assign->size-pos)
634         len = assign->size-pos;
635
636     if(this == assign) {
637         MSVCP_basic_string_char_erase(this, pos+len, MSVCP_basic_string_char_npos);
638         MSVCP_basic_string_char_erase(this, 0, pos);
639     } else if(basic_string_char_grow(this, len, FALSE)) {
640         MSVCP_char_traits_char__Copy_s(basic_string_char_ptr(this),
641                 this->res, basic_string_char_const_ptr(assign)+pos, len);
642         basic_string_char_eos(this, len);
643     }
644
645     return this;
646 }
647
648 /* ?assign@?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QAEAAV12@ABV12@@Z */
649 /* ?assign@?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QEAAAEAV12@AEBV12@@Z */
650 /* ??4?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QAEAAV01@ABV01@@Z */
651 /* ??4?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QEAAAEAV01@AEBV01@@Z */
652 DEFINE_THISCALL_WRAPPER(MSVCP_basic_string_char_assign, 8)
653 basic_string_char* __thiscall MSVCP_basic_string_char_assign(
654         basic_string_char *this, const basic_string_char *assign)
655 {
656     return MSVCP_basic_string_char_assign_substr(this, assign,
657             0, MSVCP_basic_string_char_npos);
658 }
659
660 /* ?assign@?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QAEAAV12@PBDI@Z */
661 /* ?assign@?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QEAAAEAV12@PEBD_K@Z */
662 DEFINE_THISCALL_WRAPPER(MSVCP_basic_string_char_assign_cstr_len, 12)
663 basic_string_char* __thiscall MSVCP_basic_string_char_assign_cstr_len(
664         basic_string_char *this, const char *str, MSVCP_size_t len)
665 {
666     TRACE("%p %s %lu\n", this, debugstr_a(str), len);
667
668     if(basic_string_char_inside(this, str))
669         return MSVCP_basic_string_char_assign_substr(this, this,
670                 str-basic_string_char_ptr(this), len);
671     else if(basic_string_char_grow(this, len, FALSE)) {
672         MSVCP_char_traits_char__Copy_s(basic_string_char_ptr(this),
673                 this->res, str, len);
674         basic_string_char_eos(this, len);
675     }
676
677     return this;
678 }
679
680 /* ?assign@?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QAEAAV12@PBD@Z */
681 /* ?assign@?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QEAAAEAV12@PEBD@Z */
682 /* ??4?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QAEAAV01@PBD@Z */
683 /* ??4?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QEAAAEAV01@PEBD@Z */
684 DEFINE_THISCALL_WRAPPER(MSVCP_basic_string_char_assign_cstr, 8)
685 basic_string_char* __thiscall MSVCP_basic_string_char_assign_cstr(
686         basic_string_char *this, const char *str)
687 {
688     return MSVCP_basic_string_char_assign_cstr_len(this, str,
689             MSVCP_char_traits_char_length(str));
690 }
691
692 /* ?c_str@?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QBEPBDXZ */
693 /* ?c_str@?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QEBAPEBDXZ */
694 /* ?data@?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QBEPBDXZ */
695 /* ?data@?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QEBAPEBDXZ */
696 DEFINE_THISCALL_WRAPPER(MSVCP_basic_string_char_c_str, 4)
697 const char* __thiscall MSVCP_basic_string_char_c_str(basic_string_char *this)
698 {
699     TRACE("%p\n", this);
700     return basic_string_char_const_ptr(this);
701 }
702
703 /* ?capacity@?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QBEIXZ */
704 /* ?capacity@?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QEBA_KXZ */
705 DEFINE_THISCALL_WRAPPER(MSVCP_basic_string_char_capacity, 4)
706 MSVCP_size_t __thiscall MSVCP_basic_string_char_capacity(basic_string_char *this)
707 {
708     TRACE("%p\n", this);
709     return this->res;
710 }
711
712 /* ??0?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QAE@XZ */
713 /* ??0?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QEAA@XZ */
714 DEFINE_THISCALL_WRAPPER(MSVCP_basic_string_char_ctor, 4)
715 basic_string_char* __thiscall MSVCP_basic_string_char_ctor(basic_string_char *this)
716 {
717     TRACE("%p\n", this);
718
719     basic_string_char_tidy(this, FALSE, 0);
720     return this;
721 }
722
723 /* ??0?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QAE@ABV01@@Z */
724 /* ??0?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QEAA@AEBV01@@Z */
725 DEFINE_THISCALL_WRAPPER(MSVCP_basic_string_char_copy_ctor, 8)
726 basic_string_char* __thiscall MSVCP_basic_string_char_copy_ctor(
727     basic_string_char *this, const basic_string_char *copy)
728 {
729     TRACE("%p %p\n", this, copy);
730
731     basic_string_char_tidy(this, FALSE, 0);
732     MSVCP_basic_string_char_assign(this, copy);
733     return this;
734 }
735
736 /* ??0?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QAE@PBD@Z */
737 /* ??0?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QEAA@PEBD@Z */
738 DEFINE_THISCALL_WRAPPER(MSVCP_basic_string_char_ctor_cstr, 8)
739 basic_string_char* __thiscall MSVCP_basic_string_char_ctor_cstr(
740         basic_string_char *this, const char *str)
741 {
742     TRACE("%p %s\n", this, debugstr_a(str));
743
744     basic_string_char_tidy(this, FALSE, 0);
745     MSVCP_basic_string_char_assign_cstr(this, str);
746     return this;
747 }
748
749 /* ??0?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QAE@PBDI@Z */
750 /* ??0?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QEAA@PEBD_K@Z */
751 DEFINE_THISCALL_WRAPPER(MSVCP_basic_string_char_ctor_cstr_len, 12)
752 basic_string_char* __thiscall MSVCP_basic_string_char_ctor_cstr_len(
753         basic_string_char *this, const char *str, MSVCP_size_t len)
754 {
755     TRACE("%p %s %ld\n", this, str, len);
756
757     basic_string_char_tidy(this, FALSE, 0);
758     MSVCP_basic_string_char_assign_cstr_len(this, str, len);
759     return this;
760 }
761
762 /* ??0?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QAE@ABV01@II@Z */
763 /* ??0?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QEAA@AEBV01@_K1@Z */
764 DEFINE_THISCALL_WRAPPER(MSVCP_basic_string_char_ctor_substr, 16)
765 basic_string_char* __thiscall MSVCP_basic_string_char_ctor_substr(
766         basic_string_char *this, const basic_string_char *assign,
767         MSVCP_size_t pos, MSVCP_size_t len)
768 {
769     TRACE("%p %p %lu %lu\n", this, assign, pos, len);
770
771     basic_string_char_tidy(this, FALSE, 0);
772     MSVCP_basic_string_char_assign_substr(this, assign, pos, len);
773     return this;
774 }
775
776 /* ??1?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QAE@XZ */
777 /* ??1?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QEAA@XZ */
778 DEFINE_THISCALL_WRAPPER(MSVCP_basic_string_char_dtor, 4)
779 void __thiscall MSVCP_basic_string_char_dtor(basic_string_char *this)
780 {
781     TRACE("%p\n", this);
782     basic_string_char_tidy(this, TRUE, 0);
783 }
784
785 /* ?size@?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QBEIXZ */
786 /* ?size@?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QEBA_KXZ */
787 /* ?length@?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QBEIXZ */
788 /* ?length@?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QEBA_KXZ */
789 DEFINE_THISCALL_WRAPPER(MSVCP_basic_string_char_length, 4)
790 MSVCP_size_t __thiscall MSVCP_basic_string_char_length(basic_string_char *this)
791 {
792     TRACE("%p\n", this);
793     return this->size;
794 }
795
796 /* ?swap@?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QAEXAAV12@@Z */
797 /* ?swap@?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QEAAXAEAV12@@Z */
798 DEFINE_THISCALL_WRAPPER(MSVCP_basic_string_char_swap, 8)
799 void __thiscall MSVCP_basic_string_char_swap(basic_string_char *this, basic_string_char *str)
800 {
801     if(this != str) {
802         char tmp[sizeof(this->data)];
803         const MSVCP_size_t size = this->size;
804         const MSVCP_size_t res = this->res;
805
806         memcpy(tmp, this->data.buf, sizeof(this->data));
807         memcpy(this->data.buf, str->data.buf, sizeof(this->data));
808         memcpy(str->data.buf, tmp, sizeof(this->data));
809
810         this->size = str->size;
811         this->res = str->res;
812
813         str->size = size;
814         str->res = res;
815     }
816 }
817
818 /* ?append@?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QAEAAV12@ABV12@II@Z */
819 /* ?append@?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QEAAAEAV12@AEBV12@_K1@Z */
820 DEFINE_THISCALL_WRAPPER(MSVCP_basic_string_char_append_substr, 16)
821 basic_string_char* __thiscall MSVCP_basic_string_char_append_substr(basic_string_char *this,
822         const basic_string_char *append, MSVCP_size_t offset, MSVCP_size_t count)
823 {
824     TRACE("%p %p %lu %lu\n", this, append, offset, count);
825
826     if(append->size < offset)
827         MSVCP__String_base_Xran();
828
829     if(count > append->size-offset)
830         count = append->size-offset;
831
832     if(MSVCP_basic_string_char_npos-this->size<=count || this->size+count<this->size)
833         MSVCP__String_base_Xlen();
834
835     if(basic_string_char_grow(this, this->size+count, FALSE)) {
836         MSVCP_char_traits_char__Copy_s(basic_string_char_ptr(this)+this->size,
837                 this->res-this->size, basic_string_char_const_ptr(append)+offset, count);
838         basic_string_char_eos(this, this->size+count);
839     }
840
841     return this;
842 }
843
844 /* ?append@?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QAEAAV12@ABV12@@Z */
845 /* ?append@?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QEAAAEAV12@AEBV12@@Z */
846 /* ??Y?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QAEAAV01@ABV01@@Z */
847 /* ??Y?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QEAAAEAV01@AEBV01@@Z */
848 DEFINE_THISCALL_WRAPPER(MSVCP_basic_string_char_append, 8)
849 basic_string_char* __thiscall MSVCP_basic_string_char_append(
850         basic_string_char *this, const basic_string_char *append)
851 {
852     return MSVCP_basic_string_char_append_substr(this, append,
853             0, MSVCP_basic_string_char_npos);
854 }
855
856 /* ?append@?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QAEAAV12@PBDI@Z */
857 /* ?append@?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QEAAAEAV12@PEBD_K@Z */
858 DEFINE_THISCALL_WRAPPER(MSVCP_basic_string_char_append_cstr_len, 12)
859 basic_string_char* __thiscall MSVCP_basic_string_char_append_cstr_len(
860         basic_string_char *this, const char *append, MSVCP_size_t count)
861 {
862     TRACE("%p %s %lu\n", this, append, count);
863
864     if(basic_string_char_inside(this, append))
865         return MSVCP_basic_string_char_append_substr(this, this,
866                 append-basic_string_char_ptr(this), count);
867
868     if(MSVCP_basic_string_char_npos-this->size<=count || this->size+count<this->size)
869         MSVCP__String_base_Xlen();
870
871     if(basic_string_char_grow(this, this->size+count, FALSE)) {
872         MSVCP_char_traits_char__Copy_s(basic_string_char_ptr(this)+this->size,
873                 this->res-this->size, append, count);
874         basic_string_char_eos(this, this->size+count);
875     }
876
877     return this;
878 }
879
880 /* ?append@?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QAEAAV12@PBD@Z */
881 /* ?append@?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QEAAAEAV12@PEBD@Z */
882 /* ??Y?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QAEAAV01@PBD@Z */
883 /* ??Y?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QEAAAEAV01@PEBD@Z */
884 DEFINE_THISCALL_WRAPPER(MSVCP_basic_string_char_append_cstr, 8)
885 basic_string_char* __thiscall MSVCP_basic_string_char_append_cstr(
886         basic_string_char *this, const char *append)
887 {
888     return MSVCP_basic_string_char_append_cstr_len(this, append,
889             MSVCP_char_traits_char_length(append));
890 }
891
892 /* ?append@?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QAEAAV12@ID@Z */
893 /* ?append@?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QEAAAEAV12@_KD@Z */
894 DEFINE_THISCALL_WRAPPER(MSVCP_basic_string_char_append_len_ch, 12)
895 basic_string_char* __thiscall MSVCP_basic_string_char_append_len_ch(
896         basic_string_char *this, MSVCP_size_t count, char ch)
897 {
898     TRACE("%p %lu %c\n", this, count, ch);
899
900     if(MSVCP_basic_string_char_npos-this->size <= count)
901         MSVCP__String_base_Xlen();
902
903     if(basic_string_char_grow(this, this->size+count, FALSE)) {
904         MSVCP_char_traits_char_assignn(basic_string_char_ptr(this)+this->size, count, ch);
905         basic_string_char_eos(this, this->size+count);
906     }
907
908     return this;
909 }
910
911 /* ??Y?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QAEAAV01@D@Z */
912 /* ??Y?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QEAAAEAV01@D@Z */
913 DEFINE_THISCALL_WRAPPER(MSVCP_basic_string_char_append_ch, 8)
914 basic_string_char* __thiscall MSVCP_basic_string_char_append_ch(
915         basic_string_char *this, char ch)
916 {
917     return MSVCP_basic_string_char_append_len_ch(this, 1, ch);
918 }
919
920 /* ??$?HDU?$char_traits@D@std@@V?$allocator@D@1@@std@@YA?AV?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@0@ABV10@PBD@Z */
921 /* ??$?HDU?$char_traits@D@std@@V?$allocator@D@1@@std@@YA?AV?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@0@AEBV10@PEBD@Z */
922 basic_string_char* __cdecl MSVCP_basic_string_char_concatenate_bstr_cstr(basic_string_char *ret,
923         const basic_string_char *left, const char *right)
924 {
925     TRACE("%p %s\n", left, right);
926
927     MSVCP_basic_string_char_copy_ctor(ret, left);
928     MSVCP_basic_string_char_append_cstr(ret, right);
929     return ret;
930 }
931
932 /* ??$?HDU?$char_traits@D@std@@V?$allocator@D@1@@std@@YA?AV?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@0@PBDABV10@@Z */
933 /* ??$?HDU?$char_traits@D@std@@V?$allocator@D@1@@std@@YA?AV?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@0@PEBDAEBV10@@Z */
934 basic_string_char* __cdecl MSVCP_basic_string_char_concatenate_cstr_bstr(basic_string_char *ret,
935         const char *left, const basic_string_char *right)
936 {
937     TRACE("%s %p\n", left, right);
938
939     MSVCP_basic_string_char_ctor_cstr(ret, left);
940     MSVCP_basic_string_char_append(ret, right);
941     return ret;
942 }
943
944 /* ??$?HDU?$char_traits@D@std@@V?$allocator@D@1@@std@@YA?AV?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@0@ABV10@0@Z */
945 /* ??$?HDU?$char_traits@D@std@@V?$allocator@D@1@@std@@YA?AV?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@0@AEBV10@0@Z */
946 basic_string_char* __cdecl MSVCP_basic_string_char_concatenate(basic_string_char *ret,
947         const basic_string_char *left, const basic_string_char *right)
948 {
949     TRACE("%p %p\n", left, right);
950
951     MSVCP_basic_string_char_copy_ctor(ret, left);
952     MSVCP_basic_string_char_append(ret, right);
953     return ret;
954 }
955
956 /* ?compare@?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QBEHIIPBDI@Z */
957 /* ?compare@?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QEBAH_K0PEBD0@Z */
958 DEFINE_THISCALL_WRAPPER(MSVCP_basic_string_char_compare_substr_cstr_len, 20)
959 int __thiscall MSVCP_basic_string_char_compare_substr_cstr_len(
960             const basic_string_char *this, MSVCP_size_t pos, MSVCP_size_t num,
961             const char *str, MSVCP_size_t count)
962 {
963     int ans;
964
965     TRACE("%p %lu %lu %s %lu\n", this, pos, num, str, count);
966
967     if(this->size < pos)
968         MSVCP__String_base_Xran();
969
970     if(pos+num > this->size)
971         num = this->size-pos;
972
973     ans = MSVCP_char_traits_char_compare(basic_string_char_const_ptr(this)+pos,
974             str, num>count ? count : num);
975     if(ans)
976         return ans;
977
978     if(num > count)
979         ans = 1;
980     else if(num < count)
981         ans = -1;
982     return ans;
983 }
984
985 /* ?compare@?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QBEHIIPBD@Z */
986 /* ?compare@?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QEBAH_K0PEBD@Z */
987 DEFINE_THISCALL_WRAPPER(MSVCP_basic_string_char_compare_substr_cstr, 16)
988 int __thiscall MSVCP_basic_string_char_compare_substr_cstr(const basic_string_char *this,
989         MSVCP_size_t pos, MSVCP_size_t num, const char *str)
990 {
991     return MSVCP_basic_string_char_compare_substr_cstr_len(this, pos, num,
992             str, MSVCP_char_traits_char_length(str));
993 }
994
995 /* ?compare@?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QBEHPBD@Z */
996 /* ?compare@?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QEBAHPEBD@Z */
997 DEFINE_THISCALL_WRAPPER(MSVCP_basic_string_char_compare_cstr, 8)
998 int __thiscall MSVCP_basic_string_char_compare_cstr(
999         const basic_string_char *this, const char *str)
1000 {
1001     return MSVCP_basic_string_char_compare_substr_cstr_len(this, 0, this->size,
1002             str, MSVCP_char_traits_char_length(str));
1003 }
1004
1005 /* ?compare@?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QBEHIIABV12@II@Z */
1006 /* ?compare@?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QEBAH_K0AEBV12@00@Z */
1007 DEFINE_THISCALL_WRAPPER(MSVCP_basic_string_char_compare_substr_substr, 24)
1008 int __thiscall MSVCP_basic_string_char_compare_substr_substr(
1009         const basic_string_char *this, MSVCP_size_t pos, MSVCP_size_t num,
1010         const basic_string_char *compare, MSVCP_size_t off, MSVCP_size_t count)
1011 {
1012     TRACE("%p %lu %lu %p %lu %lu\n", this, pos, num, compare, off, count);
1013
1014     if(compare->size < off)
1015         MSVCP__String_base_Xran();
1016
1017     if(off+count > compare->size)
1018         count = compare->size-off;
1019
1020     return MSVCP_basic_string_char_compare_substr_cstr_len(this, pos, num,
1021             basic_string_char_const_ptr(compare)+off, count);
1022 }
1023
1024 /* ?compare@?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QBEHIIABV12@@Z */
1025 /* ?compare@?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QEBAH_K0AEBV12@@Z */
1026 DEFINE_THISCALL_WRAPPER(MSVCP_basic_string_char_compare_substr, 16)
1027 int __thiscall MSVCP_basic_string_char_compare_substr(
1028         const basic_string_char *this, MSVCP_size_t pos, MSVCP_size_t num,
1029         const basic_string_char *compare)
1030 {
1031     return MSVCP_basic_string_char_compare_substr_cstr_len(this, pos, num,
1032             basic_string_char_const_ptr(compare), compare->size);
1033 }
1034
1035 /* ?compare@?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QBEHABV12@@Z */
1036 /* ?compare@?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QEBAHAEBV12@@Z */
1037 DEFINE_THISCALL_WRAPPER(MSVCP_basic_string_char_compare, 8)
1038 int __thiscall MSVCP_basic_string_char_compare(
1039         const basic_string_char *this, const basic_string_char *compare)
1040 {
1041     return MSVCP_basic_string_char_compare_substr_cstr_len(this, 0, this->size,
1042             basic_string_char_const_ptr(compare), compare->size);
1043 }
1044
1045 /* ??$?MDU?$char_traits@D@std@@V?$allocator@D@1@@std@@YA_NABV?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@0@0@Z */
1046 /* ??$?MDU?$char_traits@D@std@@V?$allocator@D@1@@std@@YA_NAEBV?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@0@0@Z */
1047 MSVCP_bool __cdecl MSVCP_basic_string_char_lower(
1048         const basic_string_char *left, const basic_string_char *right)
1049 {
1050     return MSVCP_basic_string_char_compare(left, right) < 0;
1051 }
1052
1053 /* ??$?MDU?$char_traits@D@std@@V?$allocator@D@1@@std@@YA_NABV?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@0@PBD@Z */
1054 /* ??$?MDU?$char_traits@D@std@@V?$allocator@D@1@@std@@YA_NAEBV?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@0@PEBD@Z */
1055 MSVCP_bool __cdecl MSVCP_basic_string_char_lower_bstr_cstr(
1056         const basic_string_char *left, const char *right)
1057 {
1058     return MSVCP_basic_string_char_compare_cstr(left, right) < 0;
1059 }
1060
1061 /* ??$?MDU?$char_traits@D@std@@V?$allocator@D@1@@std@@YA_NPBDABV?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@0@@Z */
1062 /* ??$?MDU?$char_traits@D@std@@V?$allocator@D@1@@std@@YA_NPEBDAEBV?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@0@@Z */
1063 MSVCP_bool __cdecl MSVCP_basic_string_char_lower_cstr_bstr(
1064         const char *left, const basic_string_char *right)
1065 {
1066     return MSVCP_basic_string_char_compare_cstr(right, left) > 0;
1067 }
1068
1069 /* ?find@?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QBEIPBDII@Z */
1070 /* ?find@?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QEBA_KPEBD_K1@Z */
1071 DEFINE_THISCALL_WRAPPER(MSVCP_basic_string_char_find_cstr_substr, 16)
1072 MSVCP_size_t __thiscall MSVCP_basic_string_char_find_cstr_substr(
1073         const basic_string_char *this, const char *find, MSVCP_size_t pos, MSVCP_size_t len)
1074 {
1075     const char *p, *end;
1076
1077     TRACE("%p %s %lu %lu\n", this, find, pos, len);
1078
1079     if(len==0 && pos<=this->size)
1080         return pos;
1081
1082     end = basic_string_char_const_ptr(this)+this->size-len+1;
1083     for(p=basic_string_char_const_ptr(this)+pos; p<end; p++) {
1084         p = MSVCP_char_traits_char_find(p, end-p, find);
1085         if(!p)
1086             break;
1087
1088         if(!MSVCP_char_traits_char_compare(p, find, len))
1089             return p-basic_string_char_const_ptr(this);
1090     }
1091
1092     return MSVCP_basic_string_char_npos;
1093 }
1094
1095 /* ?find@?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QBEIDI@Z */
1096 /* ?find@?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QEBA_KD_K@Z */
1097 DEFINE_THISCALL_WRAPPER(MSVCP_basic_string_char_find_ch, 12)
1098 MSVCP_size_t __thiscall MSVCP_basic_string_char_find_ch(
1099         const basic_string_char *this, char ch, MSVCP_size_t pos)
1100 {
1101     return MSVCP_basic_string_char_find_cstr_substr(this, &ch, pos, 1);
1102 }
1103
1104 /* ?at@?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QAEAADI@Z */
1105 /* ?at@?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QEAAAEAD_K@Z */
1106 /* ??A?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QAEAADI@Z */
1107 /* ??A?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QEAAAEAD_K@Z */
1108 DEFINE_THISCALL_WRAPPER(MSVCP_basic_string_char_at, 8)
1109 char* __thiscall MSVCP_basic_string_char_at(
1110         basic_string_char *this, MSVCP_size_t pos)
1111 {
1112     TRACE("%p %lu\n", this, pos);
1113
1114     if(this->size <= pos)
1115         MSVCP__String_base_Xran();
1116
1117     return basic_string_char_ptr(this)+pos;
1118 }
1119
1120 /* ?at@?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QBEABDI@Z */
1121 /* ?at@?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QEBAAEBD_K@Z */
1122 /* ??A?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QBEABDI@Z */
1123 /* ??A?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QEBAAEBD_K@Z */
1124 DEFINE_THISCALL_WRAPPER(MSVCP_basic_string_char_const_at, 8)
1125 const char* __thiscall MSVCP_basic_string_char_const_at(
1126         const basic_string_char *this, MSVCP_size_t pos)
1127 {
1128     TRACE("%p %lu\n", this, pos);
1129
1130     if(this->size <= pos)
1131         MSVCP__String_base_Xran();
1132
1133     return basic_string_char_const_ptr(this)+pos;
1134 }
1135
1136 /* ?resize@?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QAEXID@Z */
1137 /* ?resize@?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QEAAX_KD@Z */
1138 DEFINE_THISCALL_WRAPPER(MSVCP_basic_string_char_resize_ch, 12)
1139 void __thiscall MSVCP_basic_string_char_resize_ch(
1140         basic_string_char *this, MSVCP_size_t size, char ch)
1141 {
1142     TRACE("%p %lu %c\n", this, size, ch);
1143
1144     if(size <= this->size)
1145         MSVCP_basic_string_char_erase(this, size, this->size);
1146     else
1147         MSVCP_basic_string_char_append_len_ch(this, size-this->size, ch);
1148 }
1149
1150 /* ?resize@?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QAEXI@Z */
1151 /* ?resize@?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QEAAX_K@Z */
1152 DEFINE_THISCALL_WRAPPER(MSVCP_basic_string_char_resize, 8)
1153 void __thiscall MSVCP_basic_string_char_resize(
1154         basic_string_char *this, MSVCP_size_t size)
1155 {
1156     MSVCP_basic_string_char_resize_ch(this, size, '\0');
1157 }
1158
1159 /* ?clear@?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QAEXXZ */
1160 /* ?clear@?$basic_string@DU?$char_traits@D@std@@V?$allocator@D@2@@std@@QEAAXXZ */
1161 DEFINE_THISCALL_WRAPPER(MSVCP_basic_string_char_clear, 4)
1162 void __thiscall MSVCP_basic_string_char_clear(basic_string_char *this)
1163 {
1164     basic_string_char_eos(this, 0);
1165 }
1166
1167
1168 /* basic_string<wchar_t, char_traits<wchar_t>, allocator<wchar_t>> */
1169 /* ?npos@?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@std@@2IB */
1170 /* ?npos@?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@std@@2_KB */
1171 const MSVCP_size_t MSVCP_basic_string_wchar_npos = -1;
1172
1173 /* Internal: basic_string_wchar_ptr - return pointer to stored string */
1174 static wchar_t* basic_string_wchar_ptr(basic_string_wchar *this)
1175 {
1176     if(this->res == BUF_SIZE_WCHAR-1)
1177         return this->data.buf;
1178     return this->data.ptr;
1179 }
1180
1181 /* Internal: basic_string_wchar_const_ptr - returns const pointer to stored string */
1182 static const wchar_t* basic_string_wchar_const_ptr(const basic_string_wchar *this)
1183 {
1184     if(this->res == BUF_SIZE_WCHAR-1)
1185         return this->data.buf;
1186     return this->data.ptr;
1187 }
1188
1189 /* Internal: basic_string_wchar_eos - sets string length, puts '\0' on the end */
1190 static void basic_string_wchar_eos(basic_string_wchar *this, MSVCP_size_t len)
1191 {
1192     static const wchar_t nullbyte_w = '\0';
1193
1194     this->size = len;
1195     MSVCP_char_traits_wchar_assign(basic_string_wchar_ptr(this)+len, &nullbyte_w);
1196 }
1197
1198 /* Internal: basic_string_char_inside - checks if given pointer points inside stored string */
1199 static MSVCP_bool basic_string_wchar_inside(
1200         basic_string_wchar *this, const wchar_t *ptr)
1201 {
1202     wchar_t *cstr = basic_string_wchar_ptr(this);
1203
1204     return (ptr<cstr || ptr>=cstr+this->size) ? FALSE : TRUE;
1205 }
1206
1207 /* Internal: basic_string_char_tidy - initialize basic_string buffer, deallocates data */
1208 /* Caution: new_size have to be smaller than BUF_SIZE_WCHAR */
1209 static void basic_string_wchar_tidy(basic_string_wchar *this,
1210         MSVCP_bool built, MSVCP_size_t new_size)
1211 {
1212     if(built && BUF_SIZE_WCHAR<=this->res) {
1213         wchar_t *ptr = this->data.ptr;
1214
1215         if(new_size > 0)
1216             MSVCP_char_traits_wchar__Copy_s(this->data.buf, BUF_SIZE_WCHAR, ptr, new_size);
1217         MSVCP_allocator_wchar_deallocate(this->allocator, ptr, this->res+1);
1218     }
1219
1220     this->res = BUF_SIZE_WCHAR-1;
1221     basic_string_wchar_eos(this, new_size);
1222 }
1223
1224 /* Internal: basic_string_wchar_grow - changes size of internal buffer */
1225 static MSVCP_bool basic_string_wchar_grow(
1226         basic_string_wchar *this, MSVCP_size_t new_size, MSVCP_bool trim)
1227 {
1228     if(this->res < new_size) {
1229         MSVCP_size_t new_res = new_size;
1230         wchar_t *ptr;
1231
1232         new_res |= 0xf;
1233
1234         if(new_res/3 < this->res/2)
1235             new_res = this->res + this->res/2;
1236
1237         ptr = MSVCP_allocator_wchar_allocate(this->allocator, new_res);
1238         if(!ptr)
1239             ptr = MSVCP_allocator_wchar_allocate(this->allocator, new_size+1);
1240         else
1241             new_size = new_res;
1242         if(!ptr) {
1243             ERR("Out of memory\n");
1244             basic_string_wchar_tidy(this, TRUE, 0);
1245             return FALSE;
1246         }
1247
1248         MSVCP_char_traits_wchar__Copy_s(ptr, new_size,
1249                 basic_string_wchar_ptr(this), this->size);
1250         basic_string_wchar_tidy(this, TRUE, 0);
1251         this->data.ptr = ptr;
1252         this->res = new_size;
1253         basic_string_wchar_eos(this, this->size);
1254     } else if(trim && new_size < BUF_SIZE_WCHAR)
1255         basic_string_wchar_tidy(this, TRUE,
1256                 new_size<this->size ? new_size : this->size);
1257     else if(new_size == 0)
1258         basic_string_wchar_eos(this, 0);
1259
1260     return (new_size>0);
1261 }
1262
1263 /* ?erase@?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@std@@QAEAAV12@II@Z */
1264 /* ?erase@?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@std@@QEAAAEAV12@_K0@Z */
1265 DEFINE_THISCALL_WRAPPER(MSVCP_basic_string_wchar_erase, 12)
1266 basic_string_wchar* __thiscall MSVCP_basic_string_wchar_erase(
1267             basic_string_wchar *this, MSVCP_size_t pos, MSVCP_size_t len)
1268 {
1269     TRACE("%p %lu %lu\n", this, pos, len);
1270
1271     if(pos > this->size) {
1272         MSVCP__String_base_Xran();
1273         return NULL;
1274     }
1275
1276     if(len > this->size-pos)
1277         len = this->size-pos;
1278
1279     if(len) {
1280         MSVCP_char_traits_wchar__Move_s(basic_string_wchar_ptr(this)+pos,
1281                 this->res-pos, basic_string_wchar_ptr(this)+pos+len,
1282                 this->size-pos-len);
1283         basic_string_wchar_eos(this, this->size-len);
1284     }
1285
1286     return this;
1287 }
1288
1289 /* ?assign@?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@std@@QAEAAV12@ABV12@II@Z */
1290 /* ?assign@?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@std@@QEAAAEAV12@AEBV12@_K1@Z */
1291 DEFINE_THISCALL_WRAPPER(MSVCP_basic_string_wchar_assign_substr, 16)
1292 basic_string_wchar* __thiscall MSVCP_basic_string_wchar_assign_substr(
1293             basic_string_wchar *this, const basic_string_wchar *assign,
1294             MSVCP_size_t pos, MSVCP_size_t len)
1295 {
1296     TRACE("%p %p %lu %lu\n", this, assign, pos, len);
1297
1298     if(assign->size < pos) {
1299         MSVCP__String_base_Xran();
1300         return NULL;
1301     }
1302
1303     if(len > assign->size-pos)
1304         len = assign->size-pos;
1305
1306     if(this == assign) {
1307         MSVCP_basic_string_wchar_erase(this, pos+len, MSVCP_basic_string_wchar_npos);
1308         MSVCP_basic_string_wchar_erase(this, 0, pos);
1309     } else if(basic_string_wchar_grow(this, len, FALSE)) {
1310         MSVCP_char_traits_wchar__Copy_s(basic_string_wchar_ptr(this),
1311                 this->res, basic_string_wchar_const_ptr(assign)+pos, len);
1312         basic_string_wchar_eos(this, len);
1313     }
1314
1315     return this;
1316 }
1317
1318 /* ?assign@?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@std@@QAEAAV12@ABV12@@Z */
1319 /* ?assign@?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@std@@QEAAAEAV12@AEBV12@@Z */
1320 /* ??4?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@std@@QAEAAV01@ABV01@@Z */
1321 /* ??4?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@std@@QEAAAEAV01@AEBV01@@Z */
1322 DEFINE_THISCALL_WRAPPER(MSVCP_basic_string_wchar_assign, 8)
1323 basic_string_wchar* __thiscall MSVCP_basic_string_wchar_assign(
1324             basic_string_wchar *this, const basic_string_wchar *assign)
1325 {
1326     return MSVCP_basic_string_wchar_assign_substr(this, assign,
1327             0, MSVCP_basic_string_wchar_npos);
1328 }
1329
1330 /* ?assign@?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@std@@QAEAAV12@PB_WI@Z */
1331 /* ?assign@?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@std@@QEAAAEAV12@PEB_W_K@Z */
1332 DEFINE_THISCALL_WRAPPER(MSVCP_basic_string_wchar_assign_cstr_len, 12)
1333 basic_string_wchar* __thiscall MSVCP_basic_string_wchar_assign_cstr_len(
1334             basic_string_wchar *this, const wchar_t *str, MSVCP_size_t len)
1335 {
1336     TRACE("%p %s %lu\n", this, debugstr_w(str), len);
1337
1338     if(basic_string_wchar_inside(this, str))
1339         return MSVCP_basic_string_wchar_assign_substr(this, this,
1340                 str-basic_string_wchar_ptr(this), len);
1341     else if(basic_string_wchar_grow(this, len, FALSE)) {
1342         MSVCP_char_traits_wchar__Copy_s(basic_string_wchar_ptr(this),
1343                 this->res, str, len);
1344         basic_string_wchar_eos(this, len);
1345     }
1346
1347     return this;
1348 }
1349
1350 /* ?assign@?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@std@@QAEAAV12@PB_W@Z */
1351 /* ?assign@?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@std@@QEAAAEAV12@PEB_W@Z */
1352 DEFINE_THISCALL_WRAPPER(MSVCP_basic_string_wchar_assign_cstr, 8)
1353 basic_string_wchar* __thiscall MSVCP_basic_string_wchar_assign_cstr(
1354             basic_string_wchar *this, const wchar_t *str)
1355 {
1356     return MSVCP_basic_string_wchar_assign_cstr_len(this, str,
1357             MSVCP_char_traits_wchar_length(str));
1358 }
1359
1360 /* ?c_str@?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@std@@QBEPB_WXZ */
1361 /* ?c_str@?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@std@@QEBAPEB_WXZ */
1362 /* ?data@?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@std@@QBEPB_WXZ */
1363 /* ?data@?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@std@@QEBAPEB_WXZ */
1364 DEFINE_THISCALL_WRAPPER(MSVCP_basic_string_wchar_c_str, 4)
1365 const wchar_t* __thiscall MSVCP_basic_string_wchar_c_str(basic_string_wchar *this)
1366 {
1367     TRACE("%p\n", this);
1368     return basic_string_wchar_const_ptr(this);
1369 }
1370
1371 /* ?capacity@?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@std@@QBEIXZ */
1372 /* ?capacity@?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@std@@QEBA_KXZ */
1373 DEFINE_THISCALL_WRAPPER(MSVCP_basic_string_wchar_capacity, 4)
1374 MSVCP_size_t __thiscall MSVCP_basic_string_wchar_capacity(basic_string_wchar *this)
1375 {
1376     TRACE("%p\n", this);
1377     return this->res;
1378 }
1379
1380 /* ??0?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@std@@QAE@XZ */
1381 /* ??0?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@std@@QEAA@XZ */
1382 DEFINE_THISCALL_WRAPPER(MSVCP_basic_string_wchar_ctor, 4)
1383 basic_string_wchar* __thiscall MSVCP_basic_string_wchar_ctor(basic_string_wchar *this)
1384 {
1385     TRACE("%p\n", this);
1386
1387     basic_string_wchar_tidy(this, FALSE, 0);
1388     return this;
1389 }
1390
1391 /* ??0?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@std@@QAE@ABV01@@Z */
1392 /* ??0?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@std@@QEAA@AEBV01@@Z */
1393 DEFINE_THISCALL_WRAPPER(MSVCP_basic_string_wchar_copy_ctor, 8)
1394 basic_string_wchar* __thiscall MSVCP_basic_string_wchar_copy_ctor(
1395             basic_string_wchar *this, const basic_string_wchar *copy)
1396 {
1397     TRACE("%p %p\n", this, copy);
1398
1399     basic_string_wchar_tidy(this, FALSE, 0);
1400     MSVCP_basic_string_wchar_assign(this, copy);
1401     return this;
1402 }
1403
1404 /* ??0?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@std@@QAE@PB_W@Z */
1405 /* ??0?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@std@@QEAA@PEB_W@Z */
1406 DEFINE_THISCALL_WRAPPER(MSVCP_basic_string_wchar_ctor_cstr, 8)
1407 basic_string_wchar* __thiscall MSVCP_basic_string_wchar_ctor_cstr(
1408             basic_string_wchar *this, const wchar_t *str)
1409 {
1410     TRACE("%p %s\n", this, debugstr_w(str));
1411
1412     basic_string_wchar_tidy(this, FALSE, 0);
1413     MSVCP_basic_string_wchar_assign_cstr(this, str);
1414     return this;
1415 }
1416
1417 /* ??0?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@std@@QAE@PB_WI@Z */
1418 /* ??0?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@std@@QEAA@PEB_W_K@Z */
1419 DEFINE_THISCALL_WRAPPER(MSVCP_basic_string_wchar_ctor_cstr_len, 12)
1420 basic_string_wchar* __thiscall MSVCP_basic_string_wchar_ctor_cstr_len(
1421         basic_string_wchar *this, const wchar_t *str, MSVCP_size_t len)
1422 {
1423     TRACE("%p %s %ld\n", this, debugstr_w(str), len);
1424
1425     basic_string_wchar_tidy(this, FALSE, 0);
1426     MSVCP_basic_string_wchar_assign_cstr_len(this, str, len);
1427     return this;
1428 }
1429
1430 /* ??0?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@std@@QAE@ABV01@II@Z */
1431 /* ??0?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@std@@QEAA@AEBV01@_K1@Z */
1432 DEFINE_THISCALL_WRAPPER(MSVCP_basic_string_wchar_ctor_substr, 16)
1433 basic_string_wchar* __thiscall MSVCP_basic_string_wchar_ctor_substr(
1434         basic_string_wchar *this, const basic_string_wchar *assign,
1435         MSVCP_size_t pos, MSVCP_size_t len)
1436 {
1437     TRACE("%p %p %lu %lu\n", this, assign, pos, len);
1438
1439     basic_string_wchar_tidy(this, FALSE, 0);
1440     MSVCP_basic_string_wchar_assign_substr(this, assign, pos, len);
1441     return this;
1442 }
1443
1444 /* ??1?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@std@@QAE@XZ */
1445 /* ??1?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@std@@QEAA@XZ */
1446 DEFINE_THISCALL_WRAPPER(MSVCP_basic_string_wchar_dtor, 4)
1447 void __thiscall MSVCP_basic_string_wchar_dtor(basic_string_wchar *this)
1448 {
1449     TRACE("%p\n", this);
1450     basic_string_wchar_tidy(this, TRUE, 0);
1451 }
1452
1453 /* ?size@?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@std@@QBEIXZ */
1454 /* ?size@?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@std@@QEBA_KXZ */
1455 /* ?length@?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@std@@QBEIXZ */
1456 /* ?length@?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@std@@QEBA_KXZ */
1457 DEFINE_THISCALL_WRAPPER(MSVCP_basic_string_wchar_length, 4)
1458 MSVCP_size_t __thiscall MSVCP_basic_string_wchar_length(basic_string_wchar *this)
1459 {
1460     TRACE("%p\n", this);
1461     return this->size;
1462 }
1463
1464 /* ?swap@?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@std@@QAEXAAV12@@Z */
1465 /* ?swap@?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@std@@QEAAXAEAV12@@Z */
1466 DEFINE_THISCALL_WRAPPER(MSVCP_basic_string_wchar_swap, 8)
1467 void __thiscall MSVCP_basic_string_wchar_swap(basic_string_wchar *this, basic_string_wchar *str)
1468 {
1469     if(this != str) {
1470         char tmp[sizeof(this->data)];
1471         const MSVCP_size_t size = this->size;
1472         const MSVCP_size_t res = this->res;
1473
1474         memcpy(tmp, this->data.buf, sizeof(this->data));
1475         memcpy(this->data.buf, str->data.buf, sizeof(this->data));
1476         memcpy(str->data.buf, tmp, sizeof(this->data));
1477
1478         this->size = str->size;
1479         this->res = str->res;
1480
1481         str->size = size;
1482         str->res = res;
1483     }
1484 }
1485
1486 /* ?append@?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@std@@QAEAAV12@ABV12@II@Z */
1487 /* ?append@?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@std@@QEAAAEAV12@AEBV12@_K1@Z */
1488 DEFINE_THISCALL_WRAPPER(MSVCP_basic_string_wchar_append_substr, 16)
1489 basic_string_wchar* __thiscall MSVCP_basic_string_wchar_append_substr(basic_string_wchar *this,
1490         const basic_string_wchar *append, MSVCP_size_t offset, MSVCP_size_t count)
1491 {
1492     TRACE("%p %p %lu %lu\n", this, append, offset, count);
1493
1494     if(append->size < offset)
1495         MSVCP__String_base_Xran();
1496
1497     if(count > append->size-offset)
1498         count = append->size-offset;
1499
1500     if(MSVCP_basic_string_wchar_npos-this->size<=count || this->size+count<this->size)
1501         MSVCP__String_base_Xlen();
1502
1503     if(basic_string_wchar_grow(this, this->size+count, FALSE)) {
1504         MSVCP_char_traits_wchar__Copy_s(basic_string_wchar_ptr(this)+this->size,
1505                 this->res-this->size, basic_string_wchar_const_ptr(append)+offset, count);
1506         basic_string_wchar_eos(this, this->size+count);
1507     }
1508
1509     return this;
1510 }
1511
1512 /* ?append@?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@std@@QAEAAV12@ABV12@@Z */
1513 /* ?append@?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@std@@QEAAAEAV12@AEBV12@@Z */
1514 /* ??Y?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@std@@QAEAAV01@ABV01@@Z */
1515 /* ??Y?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@std@@QEAAAEAV01@AEBV01@@Z */
1516 DEFINE_THISCALL_WRAPPER(MSVCP_basic_string_wchar_append, 8)
1517 basic_string_wchar* __thiscall MSVCP_basic_string_wchar_append(
1518             basic_string_wchar *this, const basic_string_wchar *append)
1519 {
1520     return MSVCP_basic_string_wchar_append_substr(this, append,
1521             0, MSVCP_basic_string_wchar_npos);
1522 }
1523
1524 /* ?append@?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@std@@QAEAAV12@PB_WI@Z */
1525 /* ?append@?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@std@@QEAAAEAV12@PEB_W_K@Z */
1526 DEFINE_THISCALL_WRAPPER(MSVCP_basic_string_wchar_append_cstr_len, 12)
1527 basic_string_wchar* __thiscall MSVCP_basic_string_wchar_append_cstr_len(
1528         basic_string_wchar *this, const wchar_t *append, MSVCP_size_t count)
1529 {
1530     TRACE("%p %s %lu\n", this, debugstr_w(append), count);
1531
1532     if(basic_string_wchar_inside(this, append))
1533         return MSVCP_basic_string_wchar_append_substr(this, this,
1534                 append-basic_string_wchar_ptr(this), count);
1535
1536     if(MSVCP_basic_string_wchar_npos-this->size<=count || this->size+count<this->size)
1537         MSVCP__String_base_Xlen();
1538
1539     if(basic_string_wchar_grow(this, this->size+count, FALSE)) {
1540         MSVCP_char_traits_wchar__Copy_s(basic_string_wchar_ptr(this)+this->size,
1541                 this->res-this->size, append, count);
1542         basic_string_wchar_eos(this, this->size+count);
1543     }
1544
1545     return this;
1546 }
1547
1548 /* ?append@?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@std@@QAEAAV12@PB_W@Z */
1549 /* ?append@?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@std@@QEAAAEAV12@PEB_W@Z */
1550 /* ??Y?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@std@@QAEAAV01@PB_W@Z */
1551 /* ??Y?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@std@@QEAAAEAV01@PEB_W@Z */
1552 DEFINE_THISCALL_WRAPPER(MSVCP_basic_string_wchar_append_cstr, 8)
1553 basic_string_wchar* __thiscall MSVCP_basic_string_wchar_append_cstr(
1554         basic_string_wchar *this, const wchar_t *append)
1555 {
1556     return MSVCP_basic_string_wchar_append_cstr_len(this, append,
1557             MSVCP_char_traits_wchar_length(append));
1558 }
1559
1560 /* ?append@?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@std@@QAEAAV12@I_W@Z */
1561 /* ?append@?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@std@@QEAAAEAV12@_K_W@Z */
1562 DEFINE_THISCALL_WRAPPER(MSVCP_basic_string_wchar_append_len_ch, 12)
1563 basic_string_wchar* __thiscall MSVCP_basic_string_wchar_append_len_ch(
1564         basic_string_wchar *this, MSVCP_size_t count, wchar_t ch)
1565 {
1566     TRACE("%p %lu %c\n", this, count, ch);
1567
1568     if(MSVCP_basic_string_wchar_npos-this->size <= count)
1569         MSVCP__String_base_Xlen();
1570
1571     if(basic_string_wchar_grow(this, this->size+count, FALSE)) {
1572         MSVCP_char_traits_wchar_assignn(basic_string_wchar_ptr(this)+this->size, count, ch);
1573         basic_string_wchar_eos(this, this->size+count);
1574     }
1575
1576     return this;
1577 }
1578
1579 /* ??Y?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@std@@QAEAAV01@_W@Z */
1580 /* ??Y?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@std@@QEAAAEAV01@_W@Z */
1581 DEFINE_THISCALL_WRAPPER(MSVCP_basic_string_wchar_append_ch, 8)
1582 basic_string_wchar* __thiscall MSVCP_basic_string_wchar_append_ch(
1583         basic_string_wchar *this, wchar_t ch)
1584 {
1585     return MSVCP_basic_string_wchar_append_len_ch(this, 1, ch);
1586 }
1587
1588 /* ??$?H_WU?$char_traits@_W@std@@V?$allocator@_W@1@@std@@YA?AV?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@0@ABV10@PB_W@Z */
1589 /* ??$?H_WU?$char_traits@_W@std@@V?$allocator@_W@1@@std@@YA?AV?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@0@AEBV10@PEB_W@Z */
1590 basic_string_wchar* __cdecl MSVCP_basic_string_wchar_concatenate_bstr_cstr(basic_string_wchar *ret,
1591         const basic_string_wchar *left, const wchar_t *right)
1592 {
1593     TRACE("%p %s\n", left, debugstr_w(right));
1594
1595     MSVCP_basic_string_wchar_copy_ctor(ret, left);
1596     MSVCP_basic_string_wchar_append_cstr(ret, right);
1597     return ret;
1598 }
1599
1600 /* ??$?H_WU?$char_traits@_W@std@@V?$allocator@_W@1@@std@@YA?AV?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@0@PB_WABV10@@Z */
1601 /* ??$?H_WU?$char_traits@_W@std@@V?$allocator@_W@1@@std@@YA?AV?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@0@PEB_WAEBV10@@Z */
1602 basic_string_wchar* __cdecl MSVCP_basic_string_wchar_concatenate_cstr_bstr(basic_string_wchar *ret,
1603         const wchar_t *left, const basic_string_wchar *right)
1604 {
1605     TRACE("%s %p\n", debugstr_w(left), right);
1606
1607     MSVCP_basic_string_wchar_ctor_cstr(ret, left);
1608     MSVCP_basic_string_wchar_append(ret, right);
1609     return ret;
1610 }
1611
1612 /* ??$?H_WU?$char_traits@_W@std@@V?$allocator@_W@1@@std@@YA?AV?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@0@ABV10@0@Z */
1613 /* ??$?H_WU?$char_traits@_W@std@@V?$allocator@_W@1@@std@@YA?AV?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@0@AEBV10@0@Z */
1614 basic_string_wchar* __cdecl MSVCP_basic_string_wchar_concatenate(basic_string_wchar *ret,
1615         const basic_string_wchar *left, const basic_string_wchar *right)
1616 {
1617     TRACE("%p %p\n", left, right);
1618
1619     MSVCP_basic_string_wchar_copy_ctor(ret, left);
1620     MSVCP_basic_string_wchar_append(ret, right);
1621     return ret;
1622 }
1623
1624 /* ?compare@?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@std@@QBEHIIPB_WI@Z */
1625 /* ?compare@?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@std@@QEBAH_K0PEB_W0@Z */
1626 DEFINE_THISCALL_WRAPPER(MSVCP_basic_string_wchar_compare_substr_cstr_len, 20)
1627 int __thiscall MSVCP_basic_string_wchar_compare_substr_cstr_len(
1628         const basic_string_wchar *this, MSVCP_size_t pos, MSVCP_size_t num,
1629         const wchar_t *str, MSVCP_size_t count)
1630 {
1631     int ans;
1632
1633     TRACE("%p %lu %lu %s %lu\n", this, pos, num, debugstr_w(str), count);
1634
1635     if(this->size < pos)
1636         MSVCP__String_base_Xran();
1637
1638     if(pos+num > this->size)
1639         num = this->size-pos;
1640
1641     ans = MSVCP_char_traits_wchar_compare(basic_string_wchar_const_ptr(this)+pos,
1642             str, num>count ? count : num);
1643     if(ans)
1644         return ans;
1645
1646     if(num > count)
1647         ans = 1;
1648     else if(num < count)
1649         ans = -1;
1650     return ans;
1651 }
1652
1653 /* ?compare@?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@std@@QBEHIIPB_W@Z */
1654 /* ?compare@?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@std@@QEBAH_K0PEB_W@Z */
1655 DEFINE_THISCALL_WRAPPER(MSVCP_basic_string_wchar_compare_substr_cstr, 16)
1656 int __thiscall MSVCP_basic_string_wchar_compare_substr_cstr(const basic_string_wchar *this,
1657         MSVCP_size_t pos, MSVCP_size_t num, const wchar_t *str)
1658 {
1659     return MSVCP_basic_string_wchar_compare_substr_cstr_len(this, pos, num,
1660             str, MSVCP_char_traits_wchar_length(str));
1661 }
1662
1663 /* ?compare@?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@std@@QBEHPB_W@Z */
1664 /* ?compare@?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@std@@QEBAHPEB_W@Z */
1665 DEFINE_THISCALL_WRAPPER(MSVCP_basic_string_wchar_compare_cstr, 8)
1666 int __thiscall MSVCP_basic_string_wchar_compare_cstr(
1667         const basic_string_wchar *this, const wchar_t *str)
1668 {
1669     return MSVCP_basic_string_wchar_compare_substr_cstr_len(this, 0, this->size,
1670             str, MSVCP_char_traits_wchar_length(str));
1671 }
1672
1673 /* ?compare@?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@std@@QBEHIIABV12@II@Z */
1674 /* ?compare@?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@std@@QEBAH_K0AEBV12@00@Z */
1675 DEFINE_THISCALL_WRAPPER(MSVCP_basic_string_wchar_compare_substr_substr, 24)
1676 int __thiscall MSVCP_basic_string_wchar_compare_substr_substr(
1677         const basic_string_wchar *this, MSVCP_size_t pos, MSVCP_size_t num,
1678         const basic_string_wchar *compare, MSVCP_size_t off, MSVCP_size_t count)
1679 {
1680     TRACE("%p %lu %lu %p %lu %lu\n", this, pos, num, compare, off, count);
1681
1682     if(compare->size < off)
1683         MSVCP__String_base_Xran();
1684
1685     if(off+count > compare->size)
1686         count = compare->size-off;
1687
1688     return MSVCP_basic_string_wchar_compare_substr_cstr_len(this, pos, num,
1689             basic_string_wchar_const_ptr(compare)+off, count);
1690 }
1691
1692 /* ?compare@?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@std@@QBEHIIABV12@@Z */
1693 /* ?compare@?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@std@@QEBAH_K0AEBV12@@Z */
1694 DEFINE_THISCALL_WRAPPER(MSVCP_basic_string_wchar_compare_substr, 16)
1695 int __thiscall MSVCP_basic_string_wchar_compare_substr(
1696         const basic_string_wchar *this, MSVCP_size_t pos, MSVCP_size_t num,
1697         const basic_string_wchar *compare)
1698 {
1699     return MSVCP_basic_string_wchar_compare_substr_cstr_len(this, pos, num,
1700             basic_string_wchar_const_ptr(compare), compare->size);
1701 }
1702
1703 /* ?compare@?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@std@@QBEHABV12@@Z */
1704 /* ?compare@?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@std@@QEBAHAEBV12@@Z */
1705 DEFINE_THISCALL_WRAPPER(MSVCP_basic_string_wchar_compare, 8)
1706 int __thiscall MSVCP_basic_string_wchar_compare(
1707         const basic_string_wchar *this, const basic_string_wchar *compare)
1708 {
1709     return MSVCP_basic_string_wchar_compare_substr_cstr_len(this, 0, this->size,
1710             basic_string_wchar_const_ptr(compare), compare->size);
1711 }
1712
1713 /* ??$?M_WU?$char_traits@_W@std@@V?$allocator@_W@1@@std@@YA_NABV?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@0@0@Z */
1714 /* ??$?M_WU?$char_traits@_W@std@@V?$allocator@_W@1@@std@@YA_NAEBV?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@0@0@Z */
1715 MSVCP_bool __cdecl MSVCP_basic_string_wchar_lower(
1716         const basic_string_wchar *left, const basic_string_wchar *right)
1717 {
1718     return MSVCP_basic_string_wchar_compare(left, right) < 0;
1719 }
1720
1721 /* ??$?M_WU?$char_traits@_W@std@@V?$allocator@_W@1@@std@@YA_NABV?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@0@PB_W@Z */
1722 /* ??$?M_WU?$char_traits@_W@std@@V?$allocator@_W@1@@std@@YA_NAEBV?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@0@PEB_W@Z */
1723 MSVCP_bool __cdecl MSVCP_basic_string_wchar_lower_bstr_cstr(
1724         const basic_string_wchar *left, const wchar_t *right)
1725 {
1726     return MSVCP_basic_string_wchar_compare_cstr(left, right) < 0;
1727 }
1728
1729 /* ??$?M_WU?$char_traits@_W@std@@V?$allocator@_W@1@@std@@YA_NPB_WABV?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@0@@Z */
1730 /* ??$?M_WU?$char_traits@_W@std@@V?$allocator@_W@1@@std@@YA_NPEB_WAEBV?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@0@@Z */
1731 MSVCP_bool __cdecl MSVCP_basic_string_wchar_lower_cstr_bstr(
1732         const wchar_t *left, const basic_string_wchar *right)
1733 {
1734     return MSVCP_basic_string_wchar_compare_cstr(right, left) > 0;
1735 }
1736
1737 /* ?find@?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@std@@QBEIPB_WII@Z */
1738 /* ?find@?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@std@@QEBA_KPEB_W_K1@Z */
1739 DEFINE_THISCALL_WRAPPER(MSVCP_basic_string_wchar_find_cstr_substr, 16)
1740 MSVCP_size_t __thiscall MSVCP_basic_string_wchar_find_cstr_substr(
1741         const basic_string_wchar *this, const wchar_t *find, MSVCP_size_t pos, MSVCP_size_t len)
1742 {
1743     const wchar_t *p, *end;
1744
1745     TRACE("%p %s %lu %lu\n", this, debugstr_w(find), pos, len);
1746
1747     if(len==0 && pos<=this->size)
1748         return pos;
1749
1750     end = basic_string_wchar_const_ptr(this)+this->size-len+1;
1751     for(p=basic_string_wchar_const_ptr(this)+pos; p<end; p++) {
1752         p = MSVCP_char_traits_wchar_find(p, end-p, find);
1753         if(!p)
1754             break;
1755
1756         if(!MSVCP_char_traits_wchar_compare(p, find, len))
1757             return p-basic_string_wchar_const_ptr(this);
1758     }
1759
1760     return MSVCP_basic_string_wchar_npos;
1761 }
1762
1763 /* ?find@?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@std@@QBEI_WI@Z */
1764 /* ?find@?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@std@@QEBA_K_W_K@Z */
1765 DEFINE_THISCALL_WRAPPER(MSVCP_basic_string_wchar_find_ch, 12)
1766 MSVCP_size_t __thiscall MSVCP_basic_string_wchar_find_ch(
1767         const basic_string_wchar *this, wchar_t ch, MSVCP_size_t pos)
1768 {
1769     return MSVCP_basic_string_wchar_find_cstr_substr(this, &ch, pos, 1);
1770 }
1771
1772 /* ?at@?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@std@@QAEAA_WI@Z */
1773 /* ?at@?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@std@@QEAAAEA_W_K@Z */
1774 /* ??A?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@std@@QAEAA_WI@Z */
1775 /* ??A?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@std@@QEAAAEA_W_K@Z */
1776 DEFINE_THISCALL_WRAPPER(MSVCP_basic_string_wchar_at, 8)
1777 wchar_t* __thiscall MSVCP_basic_string_wchar_at(
1778         basic_string_wchar *this, MSVCP_size_t pos)
1779 {
1780     TRACE("%p %lu\n", this, pos);
1781
1782     if(this->size <= pos)
1783         MSVCP__String_base_Xran();
1784
1785     return basic_string_wchar_ptr(this)+pos;
1786 }
1787
1788 /* ?at@?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@std@@QBEAB_WI@Z */
1789 /* ?at@?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@std@@QEBAAEB_W_K@Z */
1790 /* ??A?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@std@@QBEAB_WI@Z */
1791 /* ??A?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@std@@QEBAAEB_W_K@Z */
1792 DEFINE_THISCALL_WRAPPER(MSVCP_basic_string_wchar_const_at, 8)
1793 const wchar_t* __thiscall MSVCP_basic_string_wchar_const_at(
1794         const basic_string_wchar *this, MSVCP_size_t pos)
1795 {
1796     TRACE("%p %lu\n", this, pos);
1797
1798     if(this->size <= pos)
1799         MSVCP__String_base_Xran();
1800
1801     return basic_string_wchar_const_ptr(this)+pos;
1802 }
1803
1804 /* ?resize@?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@std@@QAEXI_W@Z */
1805 /* ?resize@?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@std@@QEAAX_K_W@Z */
1806 DEFINE_THISCALL_WRAPPER(MSVCP_basic_string_wchar_resize_ch, 12)
1807 void __thiscall MSVCP_basic_string_wchar_resize_ch(
1808         basic_string_wchar *this, MSVCP_size_t size, wchar_t ch)
1809 {
1810     TRACE("%p %lu %c\n", this, size, ch);
1811
1812     if(size <= this->size)
1813         MSVCP_basic_string_wchar_erase(this, size, this->size);
1814     else
1815         MSVCP_basic_string_wchar_append_len_ch(this, size-this->size, ch);
1816 }
1817
1818 /* ?resize@?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@std@@QAEXI@Z */
1819 /* ?resize@?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@std@@QEAAX_K@Z */
1820 DEFINE_THISCALL_WRAPPER(MSVCP_basic_string_wchar_resize, 8)
1821 void __thiscall MSVCP_basic_string_wchar_resize(
1822         basic_string_wchar *this, MSVCP_size_t size)
1823 {
1824     MSVCP_basic_string_wchar_resize_ch(this, size, '\0');
1825 }
1826
1827 /* ?clear@?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@std@@QAEXXZ */
1828 /* ?clear@?$basic_string@_WU?$char_traits@_W@std@@V?$allocator@_W@2@@std@@QEAAXXZ */
1829 DEFINE_THISCALL_WRAPPER(MSVCP_basic_string_wchar_clear, 4)
1830 void __thiscall MSVCP_basic_string_wchar_clear(basic_string_wchar *this)
1831 {
1832     basic_string_wchar_eos(this, 0);
1833 }