xref: /reactos/dll/win32/shlwapi/string.c (revision 1c559240)
1 /*
2  * Shlwapi string functions
3  *
4  * Copyright 1998 Juergen Schmied
5  * Copyright 2002 Jon Griffiths
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
19  * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
20  */
21 
22 #include "config.h"
23 #include "wine/port.h"
24 
25 #include <math.h>
26 #include <stdarg.h>
27 #include <stdio.h>
28 #include <string.h>
29 
30 #define NONAMELESSUNION
31 
32 #include "windef.h"
33 #include "winbase.h"
34 #define NO_SHLWAPI_REG
35 #define NO_SHLWAPI_STREAM
36 #include "shlwapi.h"
37 #include "wingdi.h"
38 #include "winuser.h"
39 #include "shlobj.h"
40 #include "mlang.h"
41 #include "ddeml.h"
42 #include "wine/unicode.h"
43 #include "wine/debug.h"
44 
45 #include "resource.h"
46 
47 WINE_DEFAULT_DEBUG_CHANNEL(shell);
48 
49 extern HINSTANCE shlwapi_hInstance;
50 
51 static HRESULT _SHStrDupAA(LPCSTR,LPSTR*);
52 static HRESULT _SHStrDupAW(LPCWSTR,LPSTR*);
53 
54 
55 static void FillNumberFmt(NUMBERFMTW *fmt, LPWSTR decimal_buffer, int decimal_bufwlen,
56                           LPWSTR thousand_buffer, int thousand_bufwlen)
57 {
58   WCHAR grouping[64];
59   WCHAR *c;
60 
61   GetLocaleInfoW(LOCALE_USER_DEFAULT, LOCALE_ILZERO|LOCALE_RETURN_NUMBER, (LPWSTR)&fmt->LeadingZero, sizeof(fmt->LeadingZero)/sizeof(WCHAR));
62   GetLocaleInfoW(LOCALE_USER_DEFAULT, LOCALE_INEGNUMBER|LOCALE_RETURN_NUMBER, (LPWSTR)&fmt->LeadingZero, sizeof(fmt->NegativeOrder)/sizeof(WCHAR));
63   fmt->NumDigits = 0;
64   GetLocaleInfoW(LOCALE_USER_DEFAULT, LOCALE_SDECIMAL, decimal_buffer, decimal_bufwlen);
65   GetLocaleInfoW(LOCALE_USER_DEFAULT, LOCALE_STHOUSAND, thousand_buffer, thousand_bufwlen);
66   fmt->lpThousandSep = thousand_buffer;
67   fmt->lpDecimalSep = decimal_buffer;
68 
69   /*
70    * Converting grouping string to number as described on
71    * http://blogs.msdn.com/oldnewthing/archive/2006/04/18/578251.aspx
72    */
73   fmt->Grouping = 0;
74   GetLocaleInfoW(LOCALE_USER_DEFAULT, LOCALE_SGROUPING, grouping, sizeof(grouping)/sizeof(WCHAR));
75   for (c = grouping; *c; c++)
76     if (*c >= '0' && *c < '9')
77     {
78       fmt->Grouping *= 10;
79       fmt->Grouping += *c - '0';
80     }
81 
82   if (fmt->Grouping % 10 == 0)
83     fmt->Grouping /= 10;
84   else
85     fmt->Grouping *= 10;
86 }
87 
88 /*************************************************************************
89  * FormatInt   [internal]
90  *
91  * Format an integer according to the current locale
92  *
93  * RETURNS
94  *  The number of characters written on success or 0 on failure
95  */
96 static int FormatInt(LONGLONG qdwValue, LPWSTR pszBuf, int cchBuf)
97 {
98   NUMBERFMTW fmt;
99   WCHAR decimal[8], thousand[8];
100   WCHAR buf[24];
101   WCHAR *c;
102   BOOL neg = (qdwValue < 0);
103 
104   FillNumberFmt(&fmt, decimal, sizeof decimal / sizeof (WCHAR),
105                 thousand, sizeof thousand / sizeof (WCHAR));
106 
107   c = &buf[24];
108   *(--c) = 0;
109   do
110   {
111     *(--c) = '0' + (qdwValue%10);
112     qdwValue /= 10;
113   } while (qdwValue > 0);
114   if (neg)
115     *(--c) = '-';
116 
117   return GetNumberFormatW(LOCALE_USER_DEFAULT, 0, c, &fmt, pszBuf, cchBuf);
118 }
119 
120 /*************************************************************************
121  * FormatDouble   [internal]
122  *
123  * Format an integer according to the current locale. Prints the specified number of digits
124  * after the decimal point
125  *
126  * RETURNS
127  *  The number of characters written on success or 0 on failure
128  */
129 static int FormatDouble(double value, int decimals, LPWSTR pszBuf, int cchBuf)
130 {
131   static const WCHAR flfmt[] = {'%','f',0};
132   WCHAR buf[64];
133   NUMBERFMTW fmt;
134   WCHAR decimal[8], thousand[8];
135 
136   snprintfW(buf, 64, flfmt, value);
137 
138   FillNumberFmt(&fmt, decimal, sizeof decimal / sizeof (WCHAR),
139                  thousand, sizeof thousand / sizeof (WCHAR));
140   fmt.NumDigits = decimals;
141   return GetNumberFormatW(LOCALE_USER_DEFAULT, 0, buf, &fmt, pszBuf, cchBuf);
142 }
143 
144 /*************************************************************************
145  * SHLWAPI_ChrCmpHelperA
146  *
147  * Internal helper for SHLWAPI_ChrCmpA/ChrCMPIA.
148  *
149  * NOTES
150  *  Both this function and its Unicode counterpart are very inefficient. To
151  *  fix this, CompareString must be completely implemented and optimised
152  *  first. Then the core character test can be taken out of that function and
153  *  placed here, so that it need never be called at all. Until then, do not
154  *  attempt to optimise this code unless you are willing to test that it
155  *  still performs correctly.
156  */
157 static BOOL SHLWAPI_ChrCmpHelperA(WORD ch1, WORD ch2, DWORD dwFlags)
158 {
159   char str1[3], str2[3];
160 
161   str1[0] = LOBYTE(ch1);
162   if (IsDBCSLeadByte(str1[0]))
163   {
164     str1[1] = HIBYTE(ch1);
165     str1[2] = '\0';
166   }
167   else
168     str1[1] = '\0';
169 
170   str2[0] = LOBYTE(ch2);
171   if (IsDBCSLeadByte(str2[0]))
172   {
173     str2[1] = HIBYTE(ch2);
174     str2[2] = '\0';
175   }
176   else
177     str2[1] = '\0';
178 
179   return CompareStringA(GetThreadLocale(), dwFlags, str1, -1, str2, -1) - CSTR_EQUAL;
180 }
181 
182 /*************************************************************************
183  * SHLWAPI_ChrCmpA
184  *
185  * Internal helper function.
186  */
187 static BOOL WINAPI SHLWAPI_ChrCmpA(WORD ch1, WORD ch2)
188 {
189   return SHLWAPI_ChrCmpHelperA(ch1, ch2, 0);
190 }
191 
192 /*************************************************************************
193  * ChrCmpIA	(SHLWAPI.385)
194  *
195  * Compare two characters, ignoring case.
196  *
197  * PARAMS
198  *  ch1 [I] First character to compare
199  *  ch2 [I] Second character to compare
200  *
201  * RETURNS
202  *  FALSE, if the characters are equal.
203  *  Non-zero otherwise.
204  */
205 BOOL WINAPI ChrCmpIA(WORD ch1, WORD ch2)
206 {
207   TRACE("(%d,%d)\n", ch1, ch2);
208 
209   return SHLWAPI_ChrCmpHelperA(ch1, ch2, NORM_IGNORECASE);
210 }
211 
212 /*************************************************************************
213  * ChrCmpIW	[SHLWAPI.386]
214  *
215  * See ChrCmpIA.
216  */
217 BOOL WINAPI ChrCmpIW(WCHAR ch1, WCHAR ch2)
218 {
219   return CompareStringW(GetThreadLocale(), NORM_IGNORECASE, &ch1, 1, &ch2, 1) - CSTR_EQUAL;
220 }
221 
222 /*************************************************************************
223  * StrChrA	[SHLWAPI.@]
224  *
225  * Find a given character in a string.
226  *
227  * PARAMS
228  *  lpszStr [I] String to search in.
229  *  ch      [I] Character to search for.
230  *
231  * RETURNS
232  *  Success: A pointer to the first occurrence of ch in lpszStr, or NULL if
233  *           not found.
234  *  Failure: NULL, if any arguments are invalid.
235  */
236 LPSTR WINAPI StrChrA(LPCSTR lpszStr, WORD ch)
237 {
238   TRACE("(%s,%i)\n", debugstr_a(lpszStr), ch);
239 
240   if (lpszStr)
241   {
242     while (*lpszStr)
243     {
244       if (!SHLWAPI_ChrCmpA(*lpszStr, ch))
245         return (LPSTR)lpszStr;
246       lpszStr = CharNextA(lpszStr);
247     }
248   }
249   return NULL;
250 }
251 
252 /*************************************************************************
253  * StrChrW	[SHLWAPI.@]
254  *
255  * See StrChrA.
256  */
257 LPWSTR WINAPI StrChrW(LPCWSTR lpszStr, WCHAR ch)
258 {
259   LPWSTR lpszRet = NULL;
260 
261   TRACE("(%s,%i)\n", debugstr_w(lpszStr), ch);
262 
263   if (lpszStr)
264     lpszRet = strchrW(lpszStr, ch);
265   return lpszRet;
266 }
267 
268 /*************************************************************************
269  * StrChrIA	[SHLWAPI.@]
270  *
271  * Find a given character in a string, ignoring case.
272  *
273  * PARAMS
274  *  lpszStr [I] String to search in.
275  *  ch      [I] Character to search for.
276  *
277  * RETURNS
278  *  Success: A pointer to the first occurrence of ch in lpszStr, or NULL if
279  *           not found.
280  *  Failure: NULL, if any arguments are invalid.
281  */
282 LPSTR WINAPI StrChrIA(LPCSTR lpszStr, WORD ch)
283 {
284   TRACE("(%s,%i)\n", debugstr_a(lpszStr), ch);
285 
286   if (lpszStr)
287   {
288     while (*lpszStr)
289     {
290       if (!ChrCmpIA(*lpszStr, ch))
291         return (LPSTR)lpszStr;
292       lpszStr = CharNextA(lpszStr);
293     }
294   }
295   return NULL;
296 }
297 
298 /*************************************************************************
299  * StrChrIW	[SHLWAPI.@]
300  *
301  * See StrChrA.
302  */
303 LPWSTR WINAPI StrChrIW(LPCWSTR lpszStr, WCHAR ch)
304 {
305   TRACE("(%s,%i)\n", debugstr_w(lpszStr), ch);
306 
307   if (lpszStr)
308   {
309     ch = toupperW(ch);
310     while (*lpszStr)
311     {
312       if (toupperW(*lpszStr) == ch)
313         return (LPWSTR)lpszStr;
314       lpszStr++;
315     }
316     lpszStr = NULL;
317   }
318   return (LPWSTR)lpszStr;
319 }
320 
321 /*************************************************************************
322  * StrChrNW	[SHLWAPI.@]
323  */
324 LPWSTR WINAPI StrChrNW(LPCWSTR lpszStr, WCHAR ch, UINT cchMax)
325 {
326   TRACE("(%s(%i),%i)\n", debugstr_wn(lpszStr,cchMax), cchMax, ch);
327 
328   if (lpszStr)
329   {
330     while (*lpszStr && cchMax-- > 0)
331     {
332       if (*lpszStr == ch)
333         return (LPWSTR)lpszStr;
334       lpszStr++;
335     }
336   }
337   return NULL;
338 }
339 
340 /*************************************************************************
341  * StrCmpIW	[SHLWAPI.@]
342  *
343  * Compare two strings, ignoring case.
344  *
345  * PARAMS
346  *  lpszStr  [I] First string to compare
347  *  lpszComp [I] Second string to compare
348  *
349  * RETURNS
350  *  An integer less than, equal to or greater than 0, indicating that
351  *  lpszStr is less than, the same, or greater than lpszComp.
352  */
353 int WINAPI StrCmpIW(LPCWSTR lpszStr, LPCWSTR lpszComp)
354 {
355   TRACE("(%s,%s)\n", debugstr_w(lpszStr),debugstr_w(lpszComp));
356   return CompareStringW(GetThreadLocale(), NORM_IGNORECASE, lpszStr, -1, lpszComp, -1) - CSTR_EQUAL;
357 }
358 
359 /*************************************************************************
360  * StrCmpNA	[SHLWAPI.@]
361  *
362  * Compare two strings, up to a maximum length.
363  *
364  * PARAMS
365  *  lpszStr  [I] First string to compare
366  *  lpszComp [I] Second string to compare
367  *  iLen     [I] Number of chars to compare
368  *
369  * RETURNS
370  *  An integer less than, equal to or greater than 0, indicating that
371  *  lpszStr is less than, the same, or greater than lpszComp.
372  */
373 INT WINAPI StrCmpNA(LPCSTR lpszStr, LPCSTR lpszComp, INT iLen)
374 {
375   TRACE("(%s,%s,%i)\n", debugstr_a(lpszStr), debugstr_a(lpszComp), iLen);
376   return CompareStringA(GetThreadLocale(), 0, lpszStr, iLen, lpszComp, iLen) - CSTR_EQUAL;
377 }
378 
379 /*************************************************************************
380  * StrCmpNW	[SHLWAPI.@]
381  *
382  * See StrCmpNA.
383  */
384 INT WINAPI StrCmpNW(LPCWSTR lpszStr, LPCWSTR lpszComp, INT iLen)
385 {
386   TRACE("(%s,%s,%i)\n", debugstr_w(lpszStr), debugstr_w(lpszComp), iLen);
387   return CompareStringW(GetThreadLocale(), 0, lpszStr, iLen, lpszComp, iLen) - CSTR_EQUAL;
388 }
389 
390 /*************************************************************************
391  * StrCmpNIA	[SHLWAPI.@]
392  *
393  * Compare two strings, up to a maximum length, ignoring case.
394  *
395  * PARAMS
396  *  lpszStr  [I] First string to compare
397  *  lpszComp [I] Second string to compare
398  *  iLen     [I] Number of chars to compare
399  *
400  * RETURNS
401  *  An integer less than, equal to or greater than 0, indicating that
402  *  lpszStr is less than, the same, or greater than lpszComp.
403  */
404 int WINAPI StrCmpNIA(LPCSTR lpszStr, LPCSTR lpszComp, int iLen)
405 {
406   TRACE("(%s,%s,%i)\n", debugstr_a(lpszStr), debugstr_a(lpszComp), iLen);
407   return CompareStringA(GetThreadLocale(), NORM_IGNORECASE, lpszStr, iLen, lpszComp, iLen) - CSTR_EQUAL;
408 }
409 
410 /*************************************************************************
411  * StrCmpNIW	[SHLWAPI.@]
412  *
413  * See StrCmpNIA.
414  */
415 INT WINAPI StrCmpNIW(LPCWSTR lpszStr, LPCWSTR lpszComp, int iLen)
416 {
417   TRACE("(%s,%s,%i)\n", debugstr_w(lpszStr), debugstr_w(lpszComp), iLen);
418   return CompareStringW(GetThreadLocale(), NORM_IGNORECASE, lpszStr, iLen, lpszComp, iLen) - CSTR_EQUAL;
419 }
420 
421 /*************************************************************************
422  * StrCmpW	[SHLWAPI.@]
423  *
424  * Compare two strings.
425  *
426  * PARAMS
427  *  lpszStr  [I] First string to compare
428  *  lpszComp [I] Second string to compare
429  *
430  * RETURNS
431  *  An integer less than, equal to or greater than 0, indicating that
432  *  lpszStr is less than, the same, or greater than lpszComp.
433  */
434 int WINAPI StrCmpW(LPCWSTR lpszStr, LPCWSTR lpszComp)
435 {
436   TRACE("(%s,%s)\n", debugstr_w(lpszStr), debugstr_w(lpszComp));
437   return CompareStringW(GetThreadLocale(), 0, lpszStr, -1, lpszComp, -1) - CSTR_EQUAL;
438 }
439 
440 /*************************************************************************
441  * StrCatW	[SHLWAPI.@]
442  *
443  * Concatenate two strings.
444  *
445  * PARAMS
446  *  lpszStr [O] Initial string
447  *  lpszSrc [I] String to concatenate
448  *
449  * RETURNS
450  *  lpszStr.
451  */
452 LPWSTR WINAPI StrCatW(LPWSTR lpszStr, LPCWSTR lpszSrc)
453 {
454   TRACE("(%s,%s)\n", debugstr_w(lpszStr), debugstr_w(lpszSrc));
455 
456   if (lpszStr && lpszSrc)
457     strcatW(lpszStr, lpszSrc);
458   return lpszStr;
459 }
460 
461 /*************************************************************************
462  * StrCatChainW	[SHLWAPI.@]
463  *
464  * Concatenates two unicode strings.
465  *
466  * PARAMS
467  *  lpszStr [O] Initial string
468  *  cchMax  [I] Length of destination buffer
469  *  ichAt   [I] Offset from the destination buffer to begin concatenation
470  *  lpszCat [I] String to concatenate
471  *
472  * RETURNS
473  *  The offset from the beginning of pszDst to the terminating NULL.
474  */
475 DWORD WINAPI StrCatChainW(LPWSTR lpszStr, DWORD cchMax, DWORD ichAt, LPCWSTR lpszCat)
476 {
477   TRACE("(%s,%u,%d,%s)\n", debugstr_w(lpszStr), cchMax, ichAt, debugstr_w(lpszCat));
478 
479   if (ichAt == -1)
480     ichAt = strlenW(lpszStr);
481 
482   if (!cchMax)
483     return ichAt;
484 
485   if (ichAt == cchMax)
486     ichAt--;
487 
488   if (lpszCat && ichAt < cchMax)
489   {
490     lpszStr += ichAt;
491     while (ichAt < cchMax - 1 && *lpszCat)
492     {
493       *lpszStr++ = *lpszCat++;
494       ichAt++;
495     }
496     *lpszStr = 0;
497   }
498 
499   return ichAt;
500 }
501 
502 /*************************************************************************
503  * StrCpyW	[SHLWAPI.@]
504  *
505  * Copy a string to another string.
506  *
507  * PARAMS
508  *  lpszStr [O] Destination string
509  *  lpszSrc [I] Source string
510  *
511  * RETURNS
512  *  lpszStr.
513  */
514 LPWSTR WINAPI StrCpyW(LPWSTR lpszStr, LPCWSTR lpszSrc)
515 {
516   TRACE("(%p,%s)\n", lpszStr, debugstr_w(lpszSrc));
517 
518   if (lpszStr && lpszSrc)
519     strcpyW(lpszStr, lpszSrc);
520   return lpszStr;
521 }
522 
523 /*************************************************************************
524  * StrCpyNW	[SHLWAPI.@]
525  *
526  * Copy a string to another string, up to a maximum number of characters.
527  *
528  * PARAMS
529  *  dst    [O] Destination string
530  *  src    [I] Source string
531  *  count  [I] Maximum number of chars to copy
532  *
533  * RETURNS
534  *  dst.
535  */
536 LPWSTR WINAPI StrCpyNW(LPWSTR dst, LPCWSTR src, int count)
537 {
538   LPWSTR d = dst;
539   LPCWSTR s = src;
540 
541   TRACE("(%p,%s,%i)\n", dst, debugstr_w(src), count);
542 
543   if (s)
544   {
545     while ((count > 1) && *s)
546     {
547       count--;
548       *d++ = *s++;
549     }
550   }
551   if (count) *d = 0;
552 
553   return dst;
554 }
555 
556 /*************************************************************************
557  * SHLWAPI_StrStrHelperA
558  *
559  * Internal implementation of StrStrA/StrStrIA
560  */
561 static LPSTR SHLWAPI_StrStrHelperA(LPCSTR lpszStr, LPCSTR lpszSearch,
562                                    INT (WINAPI *pStrCmpFn)(LPCSTR,LPCSTR,INT))
563 {
564   size_t iLen;
565   LPCSTR end;
566 
567   if (!lpszStr || !lpszSearch || !*lpszSearch)
568     return NULL;
569 
570   iLen = strlen(lpszSearch);
571   end = lpszStr + strlen(lpszStr);
572 
573   while (lpszStr + iLen <= end)
574   {
575     if (!pStrCmpFn(lpszStr, lpszSearch, iLen))
576       return (LPSTR)lpszStr;
577     lpszStr = CharNextA(lpszStr);
578   }
579   return NULL;
580 }
581 
582 /*************************************************************************
583  * StrStrA	[SHLWAPI.@]
584  *
585  * Find a substring within a string.
586  *
587  * PARAMS
588  *  lpszStr    [I] String to search in
589  *  lpszSearch [I] String to look for
590  *
591  * RETURNS
592  *  The start of lpszSearch within lpszStr, or NULL if not found.
593  */
594 LPSTR WINAPI StrStrA(LPCSTR lpszStr, LPCSTR lpszSearch)
595 {
596   TRACE("(%s,%s)\n", debugstr_a(lpszStr), debugstr_a(lpszSearch));
597 
598   return SHLWAPI_StrStrHelperA(lpszStr, lpszSearch, StrCmpNA);
599 }
600 
601 /*************************************************************************
602  * StrStrW	[SHLWAPI.@]
603  *
604  * See StrStrA.
605  */
606 LPWSTR WINAPI StrStrW(LPCWSTR lpszStr, LPCWSTR lpszSearch)
607 {
608     TRACE("(%s, %s)\n", debugstr_w(lpszStr), debugstr_w(lpszSearch));
609 
610     if (!lpszStr || !lpszSearch || !*lpszSearch) return NULL;
611     return strstrW( lpszStr, lpszSearch );
612 }
613 
614 /*************************************************************************
615  * StrRStrIA	[SHLWAPI.@]
616  *
617  * Find the last occurrence of a substring within a string.
618  *
619  * PARAMS
620  *  lpszStr    [I] String to search in
621  *  lpszEnd    [I] End of lpszStr
622  *  lpszSearch [I] String to look for
623  *
624  * RETURNS
625  *  The last occurrence lpszSearch within lpszStr, or NULL if not found.
626  */
627 LPSTR WINAPI StrRStrIA(LPCSTR lpszStr, LPCSTR lpszEnd, LPCSTR lpszSearch)
628 {
629   LPSTR lpszRet = NULL;
630   WORD ch1, ch2;
631   INT iLen;
632 
633   TRACE("(%s,%s)\n", debugstr_a(lpszStr), debugstr_a(lpszSearch));
634 
635   if (!lpszStr || !lpszSearch || !*lpszSearch)
636     return NULL;
637 
638   if (IsDBCSLeadByte(*lpszSearch))
639     ch1 = *lpszSearch << 8 | (UCHAR)lpszSearch[1];
640   else
641     ch1 = *lpszSearch;
642   iLen = lstrlenA(lpszSearch);
643 
644   if (!lpszEnd)
645     lpszEnd = lpszStr + lstrlenA(lpszStr);
646   else /* reproduce the broken behaviour on Windows */
647     lpszEnd += min(iLen - 1, lstrlenA(lpszEnd));
648 
649   while (lpszStr + iLen <= lpszEnd && *lpszStr)
650   {
651     ch2 = IsDBCSLeadByte(*lpszStr)? *lpszStr << 8 | (UCHAR)lpszStr[1] : *lpszStr;
652     if (!ChrCmpIA(ch1, ch2))
653     {
654       if (!StrCmpNIA(lpszStr, lpszSearch, iLen))
655         lpszRet = (LPSTR)lpszStr;
656     }
657     lpszStr = CharNextA(lpszStr);
658   }
659   return lpszRet;
660 }
661 
662 /*************************************************************************
663  * StrRStrIW	[SHLWAPI.@]
664  *
665  * See StrRStrIA.
666  */
667 LPWSTR WINAPI StrRStrIW(LPCWSTR lpszStr, LPCWSTR lpszEnd, LPCWSTR lpszSearch)
668 {
669   LPWSTR lpszRet = NULL;
670   INT iLen;
671 
672   TRACE("(%s,%s)\n", debugstr_w(lpszStr), debugstr_w(lpszSearch));
673 
674   if (!lpszStr || !lpszSearch || !*lpszSearch)
675     return NULL;
676 
677   iLen = strlenW(lpszSearch);
678 
679   if (!lpszEnd)
680     lpszEnd = lpszStr + strlenW(lpszStr);
681   else /* reproduce the broken behaviour on Windows */
682     lpszEnd += min(iLen - 1, lstrlenW(lpszEnd));
683 
684   while (lpszStr + iLen <= lpszEnd && *lpszStr)
685   {
686     if (!ChrCmpIW(*lpszSearch, *lpszStr))
687     {
688       if (!StrCmpNIW(lpszStr, lpszSearch, iLen))
689         lpszRet = (LPWSTR)lpszStr;
690     }
691     lpszStr++;
692   }
693   return lpszRet;
694 }
695 
696 /*************************************************************************
697  * StrStrIA	[SHLWAPI.@]
698  *
699  * Find a substring within a string, ignoring case.
700  *
701  * PARAMS
702  *  lpszStr    [I] String to search in
703  *  lpszSearch [I] String to look for
704  *
705  * RETURNS
706  *  The start of lpszSearch within lpszStr, or NULL if not found.
707  */
708 LPSTR WINAPI StrStrIA(LPCSTR lpszStr, LPCSTR lpszSearch)
709 {
710   TRACE("(%s,%s)\n", debugstr_a(lpszStr), debugstr_a(lpszSearch));
711 
712   return SHLWAPI_StrStrHelperA(lpszStr, lpszSearch, StrCmpNIA);
713 }
714 
715 /*************************************************************************
716  * StrStrIW	[SHLWAPI.@]
717  *
718  * See StrStrIA.
719  */
720 LPWSTR WINAPI StrStrIW(LPCWSTR lpszStr, LPCWSTR lpszSearch)
721 {
722   int iLen;
723   LPCWSTR end;
724 
725   TRACE("(%s,%s)\n", debugstr_w(lpszStr), debugstr_w(lpszSearch));
726 
727   if (!lpszStr || !lpszSearch || !*lpszSearch)
728     return NULL;
729 
730   iLen = strlenW(lpszSearch);
731   end = lpszStr + strlenW(lpszStr);
732 
733   while (lpszStr + iLen <= end)
734   {
735     if (!StrCmpNIW(lpszStr, lpszSearch, iLen))
736       return (LPWSTR)lpszStr;
737     lpszStr++;
738   }
739   return NULL;
740 }
741 
742 /*************************************************************************
743  * StrStrNW	[SHLWAPI.@]
744  *
745  * Find a substring within a string up to a given number of initial characters.
746  *
747  * PARAMS
748  *  lpFirst    [I] String to search in
749  *  lpSrch     [I] String to look for
750  *  cchMax     [I] Maximum number of initial search characters
751  *
752  * RETURNS
753  *  The start of lpFirst within lpSrch, or NULL if not found.
754  */
755 LPWSTR WINAPI StrStrNW(LPCWSTR lpFirst, LPCWSTR lpSrch, UINT cchMax)
756 {
757     UINT i;
758     int len;
759 
760     TRACE("(%s, %s, %u)\n", debugstr_w(lpFirst), debugstr_w(lpSrch), cchMax);
761 
762     if (!lpFirst || !lpSrch || !*lpSrch || !cchMax)
763         return NULL;
764 
765     len = strlenW(lpSrch);
766 
767     for (i = cchMax; *lpFirst && (i > 0); i--, lpFirst++)
768     {
769         if (!strncmpW(lpFirst, lpSrch, len))
770             return (LPWSTR)lpFirst;
771     }
772 
773     return NULL;
774 }
775 
776 /*************************************************************************
777  * StrStrNIW	[SHLWAPI.@]
778  *
779  * Find a substring within a string up to a given number of initial characters,
780  * ignoring case.
781  *
782  * PARAMS
783  *  lpFirst    [I] String to search in
784  *  lpSrch     [I] String to look for
785  *  cchMax     [I] Maximum number of initial search characters
786  *
787  * RETURNS
788  *  The start of lpFirst within lpSrch, or NULL if not found.
789  */
790 LPWSTR WINAPI StrStrNIW(LPCWSTR lpFirst, LPCWSTR lpSrch, UINT cchMax)
791 {
792     UINT i;
793     int len;
794 
795     TRACE("(%s, %s, %u)\n", debugstr_w(lpFirst), debugstr_w(lpSrch), cchMax);
796 
797     if (!lpFirst || !lpSrch || !*lpSrch || !cchMax)
798         return NULL;
799 
800     len = strlenW(lpSrch);
801 
802     for (i = cchMax; *lpFirst && (i > 0); i--, lpFirst++)
803     {
804         if (!strncmpiW(lpFirst, lpSrch, len))
805             return (LPWSTR)lpFirst;
806     }
807 
808     return NULL;
809 }
810 
811 /*************************************************************************
812  * StrToIntA	[SHLWAPI.@]
813  *
814  * Read a signed integer from a string.
815  *
816  * PARAMS
817  *  lpszStr [I] String to read integer from
818  *
819  * RETURNS
820  *   The signed integer value represented by the string, or 0 if no integer is
821  *   present.
822  *
823  * NOTES
824  *  No leading space is allowed before the number, although a leading '-' is.
825  */
826 int WINAPI StrToIntA(LPCSTR lpszStr)
827 {
828   int iRet = 0;
829 
830   TRACE("(%s)\n", debugstr_a(lpszStr));
831 
832   if (!lpszStr)
833   {
834     WARN("Invalid lpszStr would crash under Win32!\n");
835     return 0;
836   }
837 
838   if (*lpszStr == '-' || isdigit(*lpszStr))
839     StrToIntExA(lpszStr, 0, &iRet);
840   return iRet;
841 }
842 
843 /*************************************************************************
844  * StrToIntW	[SHLWAPI.@]
845  *
846  * See StrToIntA.
847  */
848 int WINAPI StrToIntW(LPCWSTR lpszStr)
849 {
850   int iRet = 0;
851 
852   TRACE("(%s)\n", debugstr_w(lpszStr));
853 
854   if (!lpszStr)
855   {
856     WARN("Invalid lpszStr would crash under Win32!\n");
857     return 0;
858   }
859 
860   if (*lpszStr == '-' || isdigitW(*lpszStr))
861     StrToIntExW(lpszStr, 0, &iRet);
862   return iRet;
863 }
864 
865 /*************************************************************************
866  * StrToIntExA	[SHLWAPI.@]
867  *
868  * Read an integer from a string.
869  *
870  * PARAMS
871  *  lpszStr [I] String to read integer from
872  *  dwFlags [I] Flags controlling the conversion
873  *  lpiRet  [O] Destination for read integer.
874  *
875  * RETURNS
876  *  Success: TRUE. lpiRet contains the integer value represented by the string.
877  *  Failure: FALSE, if the string is invalid, or no number is present.
878  *
879  * NOTES
880  *  Leading whitespace, '-' and '+' are allowed before the number. If
881  *  dwFlags includes STIF_SUPPORT_HEX, hexadecimal numbers are allowed, if
882  *  preceded by '0x'. If this flag is not set, or there is no '0x' prefix,
883  *  the string is treated as a decimal string. A leading '-' is ignored for
884  *  hexadecimal numbers.
885  */
886 BOOL WINAPI StrToIntExA(LPCSTR lpszStr, DWORD dwFlags, int *lpiRet)
887 {
888   LONGLONG li;
889   BOOL bRes;
890 
891   TRACE("(%s,%08X,%p)\n", debugstr_a(lpszStr), dwFlags, lpiRet);
892 
893   bRes = StrToInt64ExA(lpszStr, dwFlags, &li);
894   if (bRes) *lpiRet = li;
895   return bRes;
896 }
897 
898 /*************************************************************************
899  * StrToInt64ExA	[SHLWAPI.@]
900  *
901  * See StrToIntExA.
902  */
903 BOOL WINAPI StrToInt64ExA(LPCSTR lpszStr, DWORD dwFlags, LONGLONG *lpiRet)
904 {
905   BOOL bNegative = FALSE;
906   LONGLONG iRet = 0;
907 
908   TRACE("(%s,%08X,%p)\n", debugstr_a(lpszStr), dwFlags, lpiRet);
909 
910   if (!lpszStr || !lpiRet)
911   {
912     WARN("Invalid parameter would crash under Win32!\n");
913     return FALSE;
914   }
915   if (dwFlags > STIF_SUPPORT_HEX) WARN("Unknown flags %08x\n", dwFlags);
916 
917   /* Skip leading space, '+', '-' */
918   while (isspace(*lpszStr))
919     lpszStr = CharNextA(lpszStr);
920 
921   if (*lpszStr == '-')
922   {
923     bNegative = TRUE;
924     lpszStr++;
925   }
926   else if (*lpszStr == '+')
927     lpszStr++;
928 
929   if (dwFlags & STIF_SUPPORT_HEX &&
930       *lpszStr == '0' && tolower(lpszStr[1]) == 'x')
931   {
932     /* Read hex number */
933     lpszStr += 2;
934 
935     if (!isxdigit(*lpszStr))
936       return FALSE;
937 
938     while (isxdigit(*lpszStr))
939     {
940       iRet = iRet * 16;
941       if (isdigit(*lpszStr))
942         iRet += (*lpszStr - '0');
943       else
944         iRet += 10 + (tolower(*lpszStr) - 'a');
945       lpszStr++;
946     }
947     *lpiRet = iRet;
948     return TRUE;
949   }
950 
951   /* Read decimal number */
952   if (!isdigit(*lpszStr))
953     return FALSE;
954 
955   while (isdigit(*lpszStr))
956   {
957     iRet = iRet * 10;
958     iRet += (*lpszStr - '0');
959     lpszStr++;
960   }
961   *lpiRet = bNegative ? -iRet : iRet;
962   return TRUE;
963 }
964 
965 /*************************************************************************
966  * StrToIntExW	[SHLWAPI.@]
967  *
968  * See StrToIntExA.
969  */
970 BOOL WINAPI StrToIntExW(LPCWSTR lpszStr, DWORD dwFlags, int *lpiRet)
971 {
972   LONGLONG li;
973   BOOL bRes;
974 
975   TRACE("(%s,%08X,%p)\n", debugstr_w(lpszStr), dwFlags, lpiRet);
976 
977   bRes = StrToInt64ExW(lpszStr, dwFlags, &li);
978   if (bRes) *lpiRet = li;
979   return bRes;
980 }
981 
982 /*************************************************************************
983  * StrToInt64ExW	[SHLWAPI.@]
984  *
985  * See StrToIntExA.
986  */
987 BOOL WINAPI StrToInt64ExW(LPCWSTR lpszStr, DWORD dwFlags, LONGLONG *lpiRet)
988 {
989   BOOL bNegative = FALSE;
990   LONGLONG iRet = 0;
991 
992   TRACE("(%s,%08X,%p)\n", debugstr_w(lpszStr), dwFlags, lpiRet);
993 
994   if (!lpszStr || !lpiRet)
995   {
996     WARN("Invalid parameter would crash under Win32!\n");
997     return FALSE;
998   }
999   if (dwFlags > STIF_SUPPORT_HEX) WARN("Unknown flags %08x\n", dwFlags);
1000 
1001   /* Skip leading space, '+', '-' */
1002   while (isspaceW(*lpszStr)) lpszStr++;
1003 
1004   if (*lpszStr == '-')
1005   {
1006     bNegative = TRUE;
1007     lpszStr++;
1008   }
1009   else if (*lpszStr == '+')
1010     lpszStr++;
1011 
1012   if (dwFlags & STIF_SUPPORT_HEX &&
1013       *lpszStr == '0' && tolowerW(lpszStr[1]) == 'x')
1014   {
1015     /* Read hex number */
1016     lpszStr += 2;
1017 
1018     if (!isxdigitW(*lpszStr))
1019       return FALSE;
1020 
1021     while (isxdigitW(*lpszStr))
1022     {
1023       iRet = iRet * 16;
1024       if (isdigitW(*lpszStr))
1025         iRet += (*lpszStr - '0');
1026       else
1027         iRet += 10 + (tolowerW(*lpszStr) - 'a');
1028       lpszStr++;
1029     }
1030     *lpiRet = iRet;
1031     return TRUE;
1032   }
1033 
1034   /* Read decimal number */
1035   if (!isdigitW(*lpszStr))
1036     return FALSE;
1037 
1038   while (isdigitW(*lpszStr))
1039   {
1040     iRet = iRet * 10;
1041     iRet += (*lpszStr - '0');
1042     lpszStr++;
1043   }
1044   *lpiRet = bNegative ? -iRet : iRet;
1045   return TRUE;
1046 }
1047 
1048 /*************************************************************************
1049  * StrDupA	[SHLWAPI.@]
1050  *
1051  * Duplicate a string.
1052  *
1053  * PARAMS
1054  *  lpszStr [I] String to duplicate.
1055  *
1056  * RETURNS
1057  *  Success: A pointer to a new string containing the contents of lpszStr
1058  *  Failure: NULL, if memory cannot be allocated
1059  *
1060  * NOTES
1061  *  The string memory is allocated with LocalAlloc(), and so should be released
1062  *  by calling LocalFree().
1063  */
1064 LPSTR WINAPI StrDupA(LPCSTR lpszStr)
1065 {
1066   int iLen;
1067   LPSTR lpszRet;
1068 
1069   TRACE("(%s)\n",debugstr_a(lpszStr));
1070 
1071 #ifdef __REACTOS__
1072   if (!lpszStr)
1073     return NULL;
1074 #endif
1075   iLen = lpszStr ? strlen(lpszStr) + 1 : 1;
1076   lpszRet = LocalAlloc(LMEM_FIXED, iLen);
1077 
1078   if (lpszRet)
1079   {
1080     if (lpszStr)
1081       memcpy(lpszRet, lpszStr, iLen);
1082     else
1083       *lpszRet = '\0';
1084   }
1085   return lpszRet;
1086 }
1087 
1088 /*************************************************************************
1089  * StrDupW	[SHLWAPI.@]
1090  *
1091  * See StrDupA.
1092  */
1093 LPWSTR WINAPI StrDupW(LPCWSTR lpszStr)
1094 {
1095   int iLen;
1096   LPWSTR lpszRet;
1097 
1098   TRACE("(%s)\n",debugstr_w(lpszStr));
1099 
1100 #ifdef __REACTOS__
1101   if (!lpszStr)
1102     return NULL;
1103 #endif
1104   iLen = (lpszStr ? strlenW(lpszStr) + 1 : 1) * sizeof(WCHAR);
1105   lpszRet = LocalAlloc(LMEM_FIXED, iLen);
1106 
1107   if (lpszRet)
1108   {
1109     if (lpszStr)
1110       memcpy(lpszRet, lpszStr, iLen);
1111     else
1112       *lpszRet = '\0';
1113   }
1114   return lpszRet;
1115 }
1116 
1117 /*************************************************************************
1118  * SHLWAPI_StrSpnHelperA
1119  *
1120  * Internal implementation of StrSpnA/StrCSpnA/StrCSpnIA
1121  */
1122 static int SHLWAPI_StrSpnHelperA(LPCSTR lpszStr, LPCSTR lpszMatch,
1123                                  LPSTR (WINAPI *pStrChrFn)(LPCSTR,WORD),
1124                                  BOOL bInvert)
1125 {
1126   LPCSTR lpszRead = lpszStr;
1127   if (lpszStr && *lpszStr && lpszMatch)
1128   {
1129     while (*lpszRead)
1130     {
1131       LPCSTR lpszTest = pStrChrFn(lpszMatch, *lpszRead);
1132 
1133       if (!bInvert && !lpszTest)
1134         break;
1135       if (bInvert && lpszTest)
1136         break;
1137       lpszRead = CharNextA(lpszRead);
1138     };
1139   }
1140   return lpszRead - lpszStr;
1141 }
1142 
1143 /*************************************************************************
1144  * StrSpnA	[SHLWAPI.@]
1145  *
1146  * Find the length of the start of a string that contains only certain
1147  * characters.
1148  *
1149  * PARAMS
1150  *  lpszStr   [I] String to search
1151  *  lpszMatch [I] Characters that can be in the substring
1152  *
1153  * RETURNS
1154  *  The length of the part of lpszStr containing only chars from lpszMatch,
1155  *  or 0 if any parameter is invalid.
1156  */
1157 int WINAPI StrSpnA(LPCSTR lpszStr, LPCSTR lpszMatch)
1158 {
1159   TRACE("(%s,%s)\n",debugstr_a(lpszStr), debugstr_a(lpszMatch));
1160 
1161   return SHLWAPI_StrSpnHelperA(lpszStr, lpszMatch, StrChrA, FALSE);
1162 }
1163 
1164 /*************************************************************************
1165  * StrSpnW	[SHLWAPI.@]
1166  *
1167  * See StrSpnA.
1168  */
1169 int WINAPI StrSpnW(LPCWSTR lpszStr, LPCWSTR lpszMatch)
1170 {
1171     if (!lpszStr || !lpszMatch) return 0;
1172     return strspnW( lpszStr, lpszMatch );
1173 }
1174 
1175 /*************************************************************************
1176  * StrCSpnA	[SHLWAPI.@]
1177  *
1178  * Find the length of the start of a string that does not contain certain
1179  * characters.
1180  *
1181  * PARAMS
1182  *  lpszStr   [I] String to search
1183  *  lpszMatch [I] Characters that cannot be in the substring
1184  *
1185  * RETURNS
1186  *  The length of the part of lpszStr containing only chars not in lpszMatch,
1187  *  or 0 if any parameter is invalid.
1188  */
1189 int WINAPI StrCSpnA(LPCSTR lpszStr, LPCSTR lpszMatch)
1190 {
1191   TRACE("(%s,%s)\n",debugstr_a(lpszStr), debugstr_a(lpszMatch));
1192 
1193   return SHLWAPI_StrSpnHelperA(lpszStr, lpszMatch, StrChrA, TRUE);
1194 }
1195 
1196 /*************************************************************************
1197  * StrCSpnW	[SHLWAPI.@]
1198  *
1199  * See StrCSpnA.
1200  */
1201 int WINAPI StrCSpnW(LPCWSTR lpszStr, LPCWSTR lpszMatch)
1202 {
1203     if (!lpszStr || !lpszMatch) return 0;
1204     return strcspnW( lpszStr, lpszMatch );
1205 }
1206 
1207 /*************************************************************************
1208  * StrCSpnIA	[SHLWAPI.@]
1209  *
1210  * Find the length of the start of a string that does not contain certain
1211  * characters, ignoring case.
1212  *
1213  * PARAMS
1214  *  lpszStr   [I] String to search
1215  *  lpszMatch [I] Characters that cannot be in the substring
1216  *
1217  * RETURNS
1218  *  The length of the part of lpszStr containing only chars not in lpszMatch,
1219  *  or 0 if any parameter is invalid.
1220  */
1221 int WINAPI StrCSpnIA(LPCSTR lpszStr, LPCSTR lpszMatch)
1222 {
1223   TRACE("(%s,%s)\n",debugstr_a(lpszStr), debugstr_a(lpszMatch));
1224 
1225   return SHLWAPI_StrSpnHelperA(lpszStr, lpszMatch, StrChrIA, TRUE);
1226 }
1227 
1228 /*************************************************************************
1229  * StrCSpnIW	[SHLWAPI.@]
1230  *
1231  * See StrCSpnIA.
1232  */
1233 int WINAPI StrCSpnIW(LPCWSTR lpszStr, LPCWSTR lpszMatch)
1234 {
1235   LPCWSTR lpszRead = lpszStr;
1236 
1237   TRACE("(%s,%s)\n",debugstr_w(lpszStr), debugstr_w(lpszMatch));
1238 
1239   if (lpszStr && *lpszStr && lpszMatch)
1240   {
1241     while (*lpszRead)
1242     {
1243       if (StrChrIW(lpszMatch, *lpszRead)) break;
1244       lpszRead++;
1245     }
1246   }
1247   return lpszRead - lpszStr;
1248 }
1249 
1250 /*************************************************************************
1251  * StrPBrkA	[SHLWAPI.@]
1252  *
1253  * Search a string for any of a group of characters.
1254  *
1255  * PARAMS
1256  *  lpszStr   [I] String to search
1257  *  lpszMatch [I] Characters to match
1258  *
1259  * RETURNS
1260  *  A pointer to the first matching character in lpszStr, or NULL if no
1261  *  match was found.
1262  */
1263 LPSTR WINAPI StrPBrkA(LPCSTR lpszStr, LPCSTR lpszMatch)
1264 {
1265   TRACE("(%s,%s)\n",debugstr_a(lpszStr), debugstr_a(lpszMatch));
1266 
1267   if (lpszStr && lpszMatch && *lpszMatch)
1268   {
1269     while (*lpszStr)
1270     {
1271       if (StrChrA(lpszMatch, *lpszStr))
1272         return (LPSTR)lpszStr;
1273       lpszStr = CharNextA(lpszStr);
1274     }
1275   }
1276   return NULL;
1277 }
1278 
1279 /*************************************************************************
1280  * StrPBrkW	[SHLWAPI.@]
1281  *
1282  * See StrPBrkA.
1283  */
1284 LPWSTR WINAPI StrPBrkW(LPCWSTR lpszStr, LPCWSTR lpszMatch)
1285 {
1286     if (!lpszStr || !lpszMatch) return NULL;
1287     return strpbrkW( lpszStr, lpszMatch );
1288 }
1289 
1290 /*************************************************************************
1291  * SHLWAPI_StrRChrHelperA
1292  *
1293  * Internal implementation of StrRChrA/StrRChrIA.
1294  */
1295 static LPSTR SHLWAPI_StrRChrHelperA(LPCSTR lpszStr,
1296                                     LPCSTR lpszEnd, WORD ch,
1297                                     BOOL (WINAPI *pChrCmpFn)(WORD,WORD))
1298 {
1299   LPCSTR lpszRet = NULL;
1300 
1301   if (lpszStr)
1302   {
1303     WORD ch2;
1304 
1305     if (!lpszEnd)
1306       lpszEnd = lpszStr + lstrlenA(lpszStr);
1307 
1308     while (*lpszStr && lpszStr <= lpszEnd)
1309     {
1310       ch2 = IsDBCSLeadByte(*lpszStr)? *lpszStr << 8 | lpszStr[1] : *lpszStr;
1311 
1312       if (!pChrCmpFn(ch, ch2))
1313         lpszRet = lpszStr;
1314       lpszStr = CharNextA(lpszStr);
1315     }
1316   }
1317   return (LPSTR)lpszRet;
1318 }
1319 
1320 /**************************************************************************
1321  * StrRChrA	[SHLWAPI.@]
1322  *
1323  * Find the last occurrence of a character in string.
1324  *
1325  * PARAMS
1326  *  lpszStr [I] String to search in
1327  *  lpszEnd [I] Place to end search, or NULL to search until the end of lpszStr
1328  *  ch      [I] Character to search for.
1329  *
1330  * RETURNS
1331  *  Success: A pointer to the last occurrence of ch in lpszStr before lpszEnd,
1332  *           or NULL if not found.
1333  *  Failure: NULL, if any arguments are invalid.
1334  */
1335 LPSTR WINAPI StrRChrA(LPCSTR lpszStr, LPCSTR lpszEnd, WORD ch)
1336 {
1337   TRACE("(%s,%s,%x)\n", debugstr_a(lpszStr), debugstr_a(lpszEnd), ch);
1338 
1339   return SHLWAPI_StrRChrHelperA(lpszStr, lpszEnd, ch, SHLWAPI_ChrCmpA);
1340 }
1341 
1342 /**************************************************************************
1343  * StrRChrW	[SHLWAPI.@]
1344  *
1345  * See StrRChrA.
1346  */
1347 LPWSTR WINAPI StrRChrW(LPCWSTR str, LPCWSTR end, WORD ch)
1348 {
1349     WCHAR *ret = NULL;
1350 
1351     if (!str) return NULL;
1352     if (!end) end = str + strlenW(str);
1353     while (str < end)
1354     {
1355         if (*str == ch) ret = (WCHAR *)str;
1356         str++;
1357     }
1358     return ret;
1359 }
1360 
1361 /**************************************************************************
1362  * StrRChrIA	[SHLWAPI.@]
1363  *
1364  * Find the last occurrence of a character in string, ignoring case.
1365  *
1366  * PARAMS
1367  *  lpszStr [I] String to search in
1368  *  lpszEnd [I] Place to end search, or NULL to search until the end of lpszStr
1369  *  ch      [I] Character to search for.
1370  *
1371  * RETURNS
1372  *  Success: A pointer to the last occurrence of ch in lpszStr before lpszEnd,
1373  *           or NULL if not found.
1374  *  Failure: NULL, if any arguments are invalid.
1375  */
1376 LPSTR WINAPI StrRChrIA(LPCSTR lpszStr, LPCSTR lpszEnd, WORD ch)
1377 {
1378   TRACE("(%s,%s,%x)\n", debugstr_a(lpszStr), debugstr_a(lpszEnd), ch);
1379 
1380   return SHLWAPI_StrRChrHelperA(lpszStr, lpszEnd, ch, ChrCmpIA);
1381 }
1382 
1383 /**************************************************************************
1384  * StrRChrIW	[SHLWAPI.@]
1385  *
1386  * See StrRChrIA.
1387  */
1388 LPWSTR WINAPI StrRChrIW(LPCWSTR str, LPCWSTR end, WORD ch)
1389 {
1390     WCHAR *ret = NULL;
1391 
1392     if (!str) return NULL;
1393     if (!end) end = str + strlenW(str);
1394     while (str < end)
1395     {
1396         if (!ChrCmpIW(*str, ch)) ret = (WCHAR *)str;
1397         str++;
1398     }
1399     return ret;
1400 }
1401 
1402 /*************************************************************************
1403  * StrCatBuffA	[SHLWAPI.@]
1404  *
1405  * Concatenate two strings together.
1406  *
1407  * PARAMS
1408  *  lpszStr [O] String to concatenate to
1409  *  lpszCat [I] String to add to lpszCat
1410  *  cchMax  [I] Maximum number of characters for the whole string
1411  *
1412  * RETURNS
1413  *  lpszStr.
1414  *
1415  * NOTES
1416  *  cchMax determines the number of characters in the final length of the
1417  *  string, not the number appended to lpszStr from lpszCat.
1418  */
1419 LPSTR WINAPI StrCatBuffA(LPSTR lpszStr, LPCSTR lpszCat, INT cchMax)
1420 {
1421   INT iLen;
1422 
1423   TRACE("(%p,%s,%d)\n", lpszStr, debugstr_a(lpszCat), cchMax);
1424 
1425   if (!lpszStr)
1426   {
1427     WARN("Invalid lpszStr would crash under Win32!\n");
1428     return NULL;
1429   }
1430 
1431   iLen = strlen(lpszStr);
1432   cchMax -= iLen;
1433 
1434   if (cchMax > 0)
1435     StrCpyNA(lpszStr + iLen, lpszCat, cchMax);
1436   return lpszStr;
1437 }
1438 
1439 /*************************************************************************
1440  * StrCatBuffW	[SHLWAPI.@]
1441  *
1442  * See StrCatBuffA.
1443  */
1444 LPWSTR WINAPI StrCatBuffW(LPWSTR lpszStr, LPCWSTR lpszCat, INT cchMax)
1445 {
1446   INT iLen;
1447 
1448   TRACE("(%p,%s,%d)\n", lpszStr, debugstr_w(lpszCat), cchMax);
1449 
1450   if (!lpszStr)
1451   {
1452     WARN("Invalid lpszStr would crash under Win32!\n");
1453     return NULL;
1454   }
1455 
1456   iLen = strlenW(lpszStr);
1457   cchMax -= iLen;
1458 
1459   if (cchMax > 0)
1460     StrCpyNW(lpszStr + iLen, lpszCat, cchMax);
1461   return lpszStr;
1462 }
1463 
1464 /*************************************************************************
1465  * StrRetToBufA					[SHLWAPI.@]
1466  *
1467  * Convert a STRRET to a normal string.
1468  *
1469  * PARAMS
1470  *  lpStrRet [O] STRRET to convert
1471  *  pIdl     [I] ITEMIDLIST for lpStrRet->uType == STRRET_OFFSET
1472  *  lpszDest [O] Destination for normal string
1473  *  dwLen    [I] Length of lpszDest
1474  *
1475  * RETURNS
1476  *  Success: S_OK. lpszDest contains up to dwLen characters of the string.
1477  *           If lpStrRet is of type STRRET_WSTR, its memory is freed with
1478  *           CoTaskMemFree() and its type set to STRRET_CSTRA.
1479  *  Failure: E_FAIL, if any parameters are invalid.
1480  */
1481 HRESULT WINAPI StrRetToBufA (LPSTRRET src, const ITEMIDLIST *pidl, LPSTR dest, UINT len)
1482 {
1483 	/* NOTE:
1484 	 *  This routine is identical to that in dlls/shell32/shellstring.c.
1485 	 *  It was duplicated because not every version of Shlwapi.dll exports
1486 	 *  StrRetToBufA. If you change one routine, change them both.
1487 	 */
1488         TRACE("dest=%p len=0x%x strret=%p pidl=%p\n", dest, len, src, pidl);
1489 
1490 	if (!src)
1491 	{
1492 	  WARN("Invalid lpStrRet would crash under Win32!\n");
1493 	  if (dest)
1494 	    *dest = '\0';
1495 	  return E_FAIL;
1496 	}
1497 
1498 	if (!dest || !len)
1499 	  return E_FAIL;
1500 
1501 	*dest = '\0';
1502 
1503 	switch (src->uType)
1504 	{
1505 	  case STRRET_WSTR:
1506 	    WideCharToMultiByte(CP_ACP, 0, src->u.pOleStr, -1, dest, len, NULL, NULL);
1507 	    CoTaskMemFree(src->u.pOleStr);
1508 	    break;
1509 
1510 	  case STRRET_CSTR:
1511             lstrcpynA(dest, src->u.cStr, len);
1512 	    break;
1513 
1514 	  case STRRET_OFFSET:
1515 	    lstrcpynA((LPSTR)dest, ((LPCSTR)&pidl->mkid)+src->u.uOffset, len);
1516 	    break;
1517 
1518 	  default:
1519 	    FIXME("unknown type!\n");
1520 	    return E_NOTIMPL;
1521 	}
1522 	return S_OK;
1523 }
1524 
1525 /*************************************************************************
1526  * StrRetToBufW	[SHLWAPI.@]
1527  *
1528  * See StrRetToBufA.
1529  */
1530 HRESULT WINAPI StrRetToBufW (LPSTRRET src, const ITEMIDLIST *pidl, LPWSTR dest, UINT len)
1531 {
1532     TRACE("dest=%p len=0x%x strret=%p pidl=%p\n", dest, len, src, pidl);
1533 
1534     if (!dest || !len)
1535         return E_FAIL;
1536 
1537     if (!src)
1538     {
1539         WARN("Invalid lpStrRet would crash under Win32!\n");
1540         if (dest)
1541             *dest = '\0';
1542         return E_FAIL;
1543     }
1544 
1545     *dest = '\0';
1546 
1547     switch (src->uType) {
1548     case STRRET_WSTR: {
1549         size_t dst_len;
1550         if (!src->u.pOleStr)
1551             return E_FAIL;
1552         dst_len = strlenW(src->u.pOleStr);
1553         memcpy(dest, src->u.pOleStr, min(dst_len, len-1) * sizeof(WCHAR));
1554         dest[min(dst_len, len-1)] = 0;
1555         CoTaskMemFree(src->u.pOleStr);
1556         if (len <= dst_len)
1557         {
1558             dest[0] = 0;
1559             return E_NOT_SUFFICIENT_BUFFER;
1560         }
1561         break;
1562     }
1563 
1564     case STRRET_CSTR:
1565         if (!MultiByteToWideChar( CP_ACP, 0, src->u.cStr, -1, dest, len ))
1566             dest[len-1] = 0;
1567         break;
1568 
1569     case STRRET_OFFSET:
1570         if (pidl)
1571 	{
1572             if (!MultiByteToWideChar( CP_ACP, 0, ((LPCSTR)&pidl->mkid)+src->u.uOffset, -1,
1573                                       dest, len ))
1574                 dest[len-1] = 0;
1575         }
1576         break;
1577 
1578     default:
1579         FIXME("unknown type!\n");
1580         return E_NOTIMPL;
1581     }
1582 
1583     return S_OK;
1584 }
1585 
1586 /*************************************************************************
1587  * StrRetToStrA					[SHLWAPI.@]
1588  *
1589  * Converts a STRRET to a normal string.
1590  *
1591  * PARAMS
1592  *  lpStrRet [O] STRRET to convert
1593  *  pidl     [I] ITEMIDLIST for lpStrRet->uType == STRRET_OFFSET
1594  *  ppszName [O] Destination for converted string
1595  *
1596  * RETURNS
1597  *  Success: S_OK. ppszName contains the new string, allocated with CoTaskMemAlloc().
1598  *  Failure: E_FAIL, if any parameters are invalid.
1599  */
1600 HRESULT WINAPI StrRetToStrA(LPSTRRET lpStrRet, const ITEMIDLIST *pidl, LPSTR *ppszName)
1601 {
1602   HRESULT hRet = E_FAIL;
1603 
1604   switch (lpStrRet->uType)
1605   {
1606   case STRRET_WSTR:
1607     hRet = _SHStrDupAW(lpStrRet->u.pOleStr, ppszName);
1608     CoTaskMemFree(lpStrRet->u.pOleStr);
1609     break;
1610 
1611   case STRRET_CSTR:
1612     hRet = _SHStrDupAA(lpStrRet->u.cStr, ppszName);
1613     break;
1614 
1615   case STRRET_OFFSET:
1616     hRet = _SHStrDupAA(((LPCSTR)&pidl->mkid) + lpStrRet->u.uOffset, ppszName);
1617     break;
1618 
1619   default:
1620     *ppszName = NULL;
1621   }
1622 
1623   return hRet;
1624 }
1625 
1626 /*************************************************************************
1627  * StrRetToStrW					[SHLWAPI.@]
1628  *
1629  * See StrRetToStrA.
1630  */
1631 HRESULT WINAPI StrRetToStrW(LPSTRRET lpStrRet, const ITEMIDLIST *pidl, LPWSTR *ppszName)
1632 {
1633   HRESULT hRet = E_FAIL;
1634 
1635   switch (lpStrRet->uType)
1636   {
1637   case STRRET_WSTR:
1638 #ifdef __REACTOS__
1639     hRet = lpStrRet->u.pOleStr ? S_OK : E_FAIL;
1640     *ppszName = lpStrRet->u.pOleStr;
1641     lpStrRet->u.pOleStr = NULL; /* Windows does this, presumably in case someone calls SHFree */
1642 #else
1643     hRet = SHStrDupW(lpStrRet->u.pOleStr, ppszName);
1644     CoTaskMemFree(lpStrRet->u.pOleStr);
1645 #endif
1646     break;
1647 
1648   case STRRET_CSTR:
1649     hRet = SHStrDupA(lpStrRet->u.cStr, ppszName);
1650     break;
1651 
1652   case STRRET_OFFSET:
1653     hRet = SHStrDupA(((LPCSTR)&pidl->mkid) + lpStrRet->u.uOffset, ppszName);
1654     break;
1655 
1656   default:
1657     *ppszName = NULL;
1658   }
1659 
1660   return hRet;
1661 }
1662 
1663 /* Create an ASCII string copy using SysAllocString() */
1664 static HRESULT _SHStrDupAToBSTR(LPCSTR src, BSTR *pBstrOut)
1665 {
1666     *pBstrOut = NULL;
1667 
1668     if (src)
1669     {
1670         INT len = MultiByteToWideChar(CP_ACP, 0, src, -1, NULL, 0);
1671         WCHAR* szTemp = HeapAlloc(GetProcessHeap(), 0, len * sizeof(WCHAR));
1672 
1673         if (szTemp)
1674         {
1675             MultiByteToWideChar(CP_ACP, 0, src, -1, szTemp, len);
1676             *pBstrOut = SysAllocString(szTemp);
1677             HeapFree(GetProcessHeap(), 0, szTemp);
1678 
1679             if (*pBstrOut)
1680                 return S_OK;
1681         }
1682     }
1683     return E_OUTOFMEMORY;
1684 }
1685 
1686 /*************************************************************************
1687  * StrRetToBSTR	[SHLWAPI.@]
1688  *
1689  * Converts a STRRET to a BSTR.
1690  *
1691  * PARAMS
1692  *  lpStrRet [O] STRRET to convert
1693  *  pidl     [I] ITEMIDLIST for lpStrRet->uType = STRRET_OFFSET
1694  *  pBstrOut [O] Destination for converted BSTR
1695  *
1696  * RETURNS
1697  *  Success: S_OK. pBstrOut contains the new string.
1698  *  Failure: E_FAIL, if any parameters are invalid.
1699  */
1700 HRESULT WINAPI StrRetToBSTR(STRRET *lpStrRet, LPCITEMIDLIST pidl, BSTR* pBstrOut)
1701 {
1702   HRESULT hRet = E_FAIL;
1703 
1704   switch (lpStrRet->uType)
1705   {
1706   case STRRET_WSTR:
1707     *pBstrOut = SysAllocString(lpStrRet->u.pOleStr);
1708     if (*pBstrOut)
1709       hRet = S_OK;
1710     CoTaskMemFree(lpStrRet->u.pOleStr);
1711     break;
1712 
1713   case STRRET_CSTR:
1714     hRet = _SHStrDupAToBSTR(lpStrRet->u.cStr, pBstrOut);
1715     break;
1716 
1717   case STRRET_OFFSET:
1718     hRet = _SHStrDupAToBSTR(((LPCSTR)&pidl->mkid) + lpStrRet->u.uOffset, pBstrOut);
1719     break;
1720 
1721   default:
1722     *pBstrOut = NULL;
1723   }
1724 
1725   return hRet;
1726 }
1727 
1728 /*************************************************************************
1729  * StrFormatKBSizeA	[SHLWAPI.@]
1730  *
1731  * Create a formatted string containing a byte count in Kilobytes.
1732  *
1733  * PARAMS
1734  *  llBytes  [I] Byte size to format
1735  *  lpszDest [I] Destination for formatted string
1736  *  cchMax   [I] Size of lpszDest
1737  *
1738  * RETURNS
1739  *  lpszDest.
1740  */
1741 LPSTR WINAPI StrFormatKBSizeA(LONGLONG llBytes, LPSTR lpszDest, UINT cchMax)
1742 {
1743   WCHAR wszBuf[256];
1744 
1745   if (!StrFormatKBSizeW(llBytes, wszBuf, 256))
1746     return NULL;
1747   if (!WideCharToMultiByte(CP_ACP, 0, wszBuf, -1, lpszDest, cchMax, NULL, NULL))
1748     return NULL;
1749   return lpszDest;
1750 }
1751 
1752 /*************************************************************************
1753  * StrFormatKBSizeW	[SHLWAPI.@]
1754  *
1755  * See StrFormatKBSizeA.
1756  */
1757 LPWSTR WINAPI StrFormatKBSizeW(LONGLONG llBytes, LPWSTR lpszDest, UINT cchMax)
1758 {
1759   static const WCHAR kb[] = {' ','K','B',0};
1760   LONGLONG llKB = (llBytes + 1023) >> 10;
1761   int len;
1762 
1763   TRACE("(0x%s,%p,%d)\n", wine_dbgstr_longlong(llBytes), lpszDest, cchMax);
1764 
1765   if (!FormatInt(llKB, lpszDest, cchMax))
1766     return NULL;
1767 
1768   len = lstrlenW(lpszDest);
1769   if (cchMax - len < 4)
1770       return NULL;
1771   lstrcatW(lpszDest, kb);
1772   return lpszDest;
1773 }
1774 
1775 /*************************************************************************
1776  * StrNCatA	[SHLWAPI.@]
1777  *
1778  * Concatenate two strings together.
1779  *
1780  * PARAMS
1781  *  lpszStr [O] String to concatenate to
1782  *  lpszCat [I] String to add to lpszCat
1783  *  cchMax  [I] Maximum number of characters to concatenate
1784  *
1785  * RETURNS
1786  *  lpszStr.
1787  *
1788  * NOTES
1789  *  cchMax determines the number of characters that are appended to lpszStr,
1790  *  not the total length of the string.
1791  */
1792 LPSTR WINAPI StrNCatA(LPSTR lpszStr, LPCSTR lpszCat, INT cchMax)
1793 {
1794   LPSTR lpszRet = lpszStr;
1795 
1796   TRACE("(%s,%s,%i)\n", debugstr_a(lpszStr), debugstr_a(lpszCat), cchMax);
1797 
1798   if (!lpszStr)
1799   {
1800     WARN("Invalid lpszStr would crash under Win32!\n");
1801     return NULL;
1802   }
1803 
1804   StrCpyNA(lpszStr + strlen(lpszStr), lpszCat, cchMax);
1805   return lpszRet;
1806 }
1807 
1808 /*************************************************************************
1809  * StrNCatW	[SHLWAPI.@]
1810  *
1811  * See StrNCatA.
1812  */
1813 LPWSTR WINAPI StrNCatW(LPWSTR lpszStr, LPCWSTR lpszCat, INT cchMax)
1814 {
1815   LPWSTR lpszRet = lpszStr;
1816 
1817   TRACE("(%s,%s,%i)\n", debugstr_w(lpszStr), debugstr_w(lpszCat), cchMax);
1818 
1819   if (!lpszStr)
1820   {
1821     WARN("Invalid lpszStr would crash under Win32\n");
1822     return NULL;
1823   }
1824 
1825   StrCpyNW(lpszStr + strlenW(lpszStr), lpszCat, cchMax);
1826   return lpszRet;
1827 }
1828 
1829 /*************************************************************************
1830  * StrTrimA	[SHLWAPI.@]
1831  *
1832  * Remove characters from the start and end of a string.
1833  *
1834  * PARAMS
1835  *  lpszStr  [O] String to remove characters from
1836  *  lpszTrim [I] Characters to remove from lpszStr
1837  *
1838  * RETURNS
1839  *  TRUE  If lpszStr was valid and modified
1840  *  FALSE Otherwise
1841  */
1842 BOOL WINAPI StrTrimA(LPSTR lpszStr, LPCSTR lpszTrim)
1843 {
1844   DWORD dwLen;
1845   LPSTR lpszRead = lpszStr;
1846   BOOL bRet = FALSE;
1847 
1848   TRACE("(%s,%s)\n", debugstr_a(lpszStr), debugstr_a(lpszTrim));
1849 
1850   if (lpszRead && *lpszRead)
1851   {
1852     while (*lpszRead && StrChrA(lpszTrim, *lpszRead))
1853       lpszRead = CharNextA(lpszRead); /* Skip leading matches */
1854 
1855     dwLen = strlen(lpszRead);
1856 
1857     if (lpszRead != lpszStr)
1858     {
1859       memmove(lpszStr, lpszRead, dwLen + 1);
1860       bRet = TRUE;
1861     }
1862     if (dwLen > 0)
1863     {
1864       lpszRead = lpszStr + dwLen;
1865       while (StrChrA(lpszTrim, lpszRead[-1]))
1866         lpszRead = CharPrevA(lpszStr, lpszRead); /* Skip trailing matches */
1867 
1868       if (lpszRead != lpszStr + dwLen)
1869       {
1870         *lpszRead = '\0';
1871         bRet = TRUE;
1872       }
1873     }
1874   }
1875   return bRet;
1876 }
1877 
1878 /*************************************************************************
1879  * StrTrimW	[SHLWAPI.@]
1880  *
1881  * See StrTrimA.
1882  */
1883 BOOL WINAPI StrTrimW(LPWSTR lpszStr, LPCWSTR lpszTrim)
1884 {
1885   DWORD dwLen;
1886   LPWSTR lpszRead = lpszStr;
1887   BOOL bRet = FALSE;
1888 
1889   TRACE("(%s,%s)\n", debugstr_w(lpszStr), debugstr_w(lpszTrim));
1890 
1891   if (lpszRead && *lpszRead)
1892   {
1893     while (*lpszRead && StrChrW(lpszTrim, *lpszRead)) lpszRead++;
1894 
1895     dwLen = strlenW(lpszRead);
1896 
1897     if (lpszRead != lpszStr)
1898     {
1899       memmove(lpszStr, lpszRead, (dwLen + 1) * sizeof(WCHAR));
1900       bRet = TRUE;
1901     }
1902     if (dwLen > 0)
1903     {
1904       lpszRead = lpszStr + dwLen;
1905       while (StrChrW(lpszTrim, lpszRead[-1]))
1906         lpszRead--; /* Skip trailing matches */
1907 
1908       if (lpszRead != lpszStr + dwLen)
1909       {
1910         *lpszRead = '\0';
1911         bRet = TRUE;
1912       }
1913     }
1914   }
1915   return bRet;
1916 }
1917 
1918 /*************************************************************************
1919  *      _SHStrDupAA	[INTERNAL]
1920  *
1921  * Duplicates a ASCII string to ASCII. The destination buffer is allocated.
1922  */
1923 static HRESULT _SHStrDupAA(LPCSTR src, LPSTR * dest)
1924 {
1925 	HRESULT hr;
1926 	int len = 0;
1927 
1928 	if (src) {
1929 	    len = lstrlenA(src) + 1;
1930 	    *dest = CoTaskMemAlloc(len);
1931 	} else {
1932 	    *dest = NULL;
1933 	}
1934 
1935 	if (*dest) {
1936 	    lstrcpynA(*dest,src, len);
1937 	    hr = S_OK;
1938 	} else {
1939 	    hr = E_OUTOFMEMORY;
1940 	}
1941 
1942 	TRACE("%s->(%p)\n", debugstr_a(src), *dest);
1943 	return hr;
1944 }
1945 
1946 /*************************************************************************
1947  * SHStrDupA	[SHLWAPI.@]
1948  *
1949  * Return a Unicode copy of a string, in memory allocated by CoTaskMemAlloc().
1950  *
1951  * PARAMS
1952  *  lpszStr   [I] String to copy
1953  *  lppszDest [O] Destination for the new string copy
1954  *
1955  * RETURNS
1956  *  Success: S_OK. lppszDest contains the new string in Unicode format.
1957  *  Failure: E_OUTOFMEMORY, If any arguments are invalid or memory allocation
1958  *           fails.
1959  */
1960 HRESULT WINAPI SHStrDupA(LPCSTR lpszStr, LPWSTR * lppszDest)
1961 {
1962   HRESULT hRet;
1963   int len = 0;
1964 
1965   if (lpszStr)
1966   {
1967     len = MultiByteToWideChar(CP_ACP, 0, lpszStr, -1, NULL, 0) * sizeof(WCHAR);
1968     *lppszDest = CoTaskMemAlloc(len);
1969   }
1970   else
1971     *lppszDest = NULL;
1972 
1973   if (*lppszDest)
1974   {
1975     MultiByteToWideChar(CP_ACP, 0, lpszStr, -1, *lppszDest, len/sizeof(WCHAR));
1976     hRet = S_OK;
1977   }
1978   else
1979     hRet = E_OUTOFMEMORY;
1980 
1981   TRACE("%s->(%p)\n", debugstr_a(lpszStr), *lppszDest);
1982   return hRet;
1983 }
1984 
1985 /*************************************************************************
1986  *      _SHStrDupAW	[INTERNAL]
1987  *
1988  * Duplicates a UNICODE to a ASCII string. The destination buffer is allocated.
1989  */
1990 static HRESULT _SHStrDupAW(LPCWSTR src, LPSTR * dest)
1991 {
1992 	HRESULT hr;
1993 	int len = 0;
1994 
1995 	if (src) {
1996 	    len = WideCharToMultiByte(CP_ACP, 0, src, -1, NULL, 0, NULL, NULL);
1997 	    *dest = CoTaskMemAlloc(len);
1998 	} else {
1999 	    *dest = NULL;
2000 	}
2001 
2002 	if (*dest) {
2003 	    WideCharToMultiByte(CP_ACP, 0, src, -1, *dest, len, NULL, NULL);
2004 	    hr = S_OK;
2005 	} else {
2006 	    hr = E_OUTOFMEMORY;
2007 	}
2008 
2009 	TRACE("%s->(%p)\n", debugstr_w(src), *dest);
2010 	return hr;
2011 }
2012 
2013 /*************************************************************************
2014  * SHStrDupW	[SHLWAPI.@]
2015  *
2016  * See SHStrDupA.
2017  */
2018 HRESULT WINAPI SHStrDupW(LPCWSTR src, LPWSTR * dest)
2019 {
2020 	HRESULT hr;
2021 	int len = 0;
2022 
2023 	if (src) {
2024 	    len = (lstrlenW(src) + 1) * sizeof(WCHAR);
2025 	    *dest = CoTaskMemAlloc(len);
2026 	} else {
2027 	    *dest = NULL;
2028 	}
2029 
2030 	if (*dest) {
2031 	    memcpy(*dest, src, len);
2032 	    hr = S_OK;
2033 	} else {
2034 	    hr = E_OUTOFMEMORY;
2035 	}
2036 
2037 	TRACE("%s->(%p)\n", debugstr_w(src), *dest);
2038 	return hr;
2039 }
2040 
2041 /*************************************************************************
2042  * SHLWAPI_WriteReverseNum
2043  *
2044  * Internal helper for SHLWAPI_WriteTimeClass.
2045  */
2046 static inline LPWSTR SHLWAPI_WriteReverseNum(LPWSTR lpszOut, DWORD dwNum)
2047 {
2048   *lpszOut-- = '\0';
2049 
2050   /* Write a decimal number to a string, backwards */
2051   do
2052   {
2053     DWORD dwNextDigit = dwNum % 10;
2054     *lpszOut-- = '0' + dwNextDigit;
2055     dwNum = (dwNum - dwNextDigit) / 10;
2056   } while (dwNum > 0);
2057 
2058   return lpszOut;
2059 }
2060 
2061 /*************************************************************************
2062  * SHLWAPI_FormatSignificant
2063  *
2064  * Internal helper for SHLWAPI_WriteTimeClass.
2065  */
2066 static inline int SHLWAPI_FormatSignificant(LPWSTR lpszNum, int dwDigits)
2067 {
2068   /* Zero non significant digits, return remaining significant digits */
2069   while (*lpszNum)
2070   {
2071     lpszNum++;
2072     if (--dwDigits == 0)
2073     {
2074       while (*lpszNum)
2075         *lpszNum++ = '0';
2076       return 0;
2077     }
2078   }
2079   return dwDigits;
2080 }
2081 
2082 /*************************************************************************
2083  * SHLWAPI_WriteTimeClass
2084  *
2085  * Internal helper for StrFromTimeIntervalW.
2086  */
2087 static int SHLWAPI_WriteTimeClass(LPWSTR lpszOut, DWORD dwValue,
2088                                   UINT uClassStringId, int iDigits)
2089 {
2090   WCHAR szBuff[64], *szOut = szBuff + 32;
2091 
2092   szOut = SHLWAPI_WriteReverseNum(szOut, dwValue);
2093   iDigits = SHLWAPI_FormatSignificant(szOut + 1, iDigits);
2094   *szOut = ' ';
2095   LoadStringW(shlwapi_hInstance, uClassStringId, szBuff + 32, 32);
2096   strcatW(lpszOut, szOut);
2097   return iDigits;
2098 }
2099 
2100 /*************************************************************************
2101  * StrFromTimeIntervalA	[SHLWAPI.@]
2102  *
2103  * Format a millisecond time interval into a string
2104  *
2105  * PARAMS
2106  *  lpszStr  [O] Output buffer for formatted time interval
2107  *  cchMax   [I] Size of lpszStr
2108  *  dwMS     [I] Number of milliseconds
2109  *  iDigits  [I] Number of digits to print
2110  *
2111  * RETURNS
2112  *  The length of the formatted string, or 0 if any parameter is invalid.
2113  *
2114  * NOTES
2115  *  This implementation mimics the Win32 behaviour of always writing a leading
2116  *  space before the time interval begins.
2117  *
2118  *  iDigits is used to provide approximate times if accuracy is not important.
2119  *  This number of digits will be written of the first non-zero time class
2120  *  (hours/minutes/seconds). If this does not complete the time classification,
2121  *  the remaining digits are changed to zeros (i.e. The time is _not_ rounded).
2122  *  If there are digits remaining following the writing of a time class, the
2123  *  next time class will be written.
2124  *
2125  *  For example, given dwMS represents 138 hours,43 minutes and 15 seconds, the
2126  *  following will result from the given values of iDigits:
2127  *
2128  *|  iDigits    1        2        3        4               5               ...
2129  *|  lpszStr   "100 hr" "130 hr" "138 hr" "138 hr 40 min" "138 hr 43 min"  ...
2130  */
2131 INT WINAPI StrFromTimeIntervalA(LPSTR lpszStr, UINT cchMax, DWORD dwMS,
2132                                 int iDigits)
2133 {
2134   INT iRet = 0;
2135 
2136   TRACE("(%p,%d,%d,%d)\n", lpszStr, cchMax, dwMS, iDigits);
2137 
2138   if (lpszStr && cchMax)
2139   {
2140     WCHAR szBuff[128];
2141     StrFromTimeIntervalW(szBuff, sizeof(szBuff)/sizeof(WCHAR), dwMS, iDigits);
2142     WideCharToMultiByte(CP_ACP,0,szBuff,-1,lpszStr,cchMax,0,0);
2143   }
2144   return iRet;
2145 }
2146 
2147 
2148 /*************************************************************************
2149  * StrFromTimeIntervalW	[SHLWAPI.@]
2150  *
2151  * See StrFromTimeIntervalA.
2152  */
2153 INT WINAPI StrFromTimeIntervalW(LPWSTR lpszStr, UINT cchMax, DWORD dwMS,
2154                                 int iDigits)
2155 {
2156   INT iRet = 0;
2157 
2158   TRACE("(%p,%d,%d,%d)\n", lpszStr, cchMax, dwMS, iDigits);
2159 
2160   if (lpszStr && cchMax)
2161   {
2162     WCHAR szCopy[128];
2163     DWORD dwHours, dwMinutes;
2164 
2165     if (!iDigits || cchMax == 1)
2166     {
2167       *lpszStr = '\0';
2168       return 0;
2169     }
2170 
2171     /* Calculate the time classes */
2172     dwMS = (dwMS + 500) / 1000;
2173     dwHours = dwMS / 3600;
2174     dwMS -= dwHours * 3600;
2175     dwMinutes = dwMS / 60;
2176     dwMS -= dwMinutes * 60;
2177 
2178     szCopy[0] = '\0';
2179 
2180     if (dwHours)
2181       iDigits = SHLWAPI_WriteTimeClass(szCopy, dwHours, IDS_TIME_INTERVAL_HOURS, iDigits);
2182 
2183     if (dwMinutes && iDigits)
2184       iDigits = SHLWAPI_WriteTimeClass(szCopy, dwMinutes, IDS_TIME_INTERVAL_MINUTES, iDigits);
2185 
2186     if (iDigits) /* Always write seconds if we have significant digits */
2187       SHLWAPI_WriteTimeClass(szCopy, dwMS, IDS_TIME_INTERVAL_SECONDS, iDigits);
2188 
2189     lstrcpynW(lpszStr, szCopy, cchMax);
2190     iRet = strlenW(lpszStr);
2191   }
2192   return iRet;
2193 }
2194 
2195 /*************************************************************************
2196  * StrIsIntlEqualA	[SHLWAPI.@]
2197  *
2198  * Compare two strings.
2199  *
2200  * PARAMS
2201  *  bCase    [I] Whether to compare case sensitively
2202  *  lpszStr  [I] First string to compare
2203  *  lpszComp [I] Second string to compare
2204  *  iLen     [I] Length to compare
2205  *
2206  * RETURNS
2207  *  TRUE  If the strings are equal.
2208  *  FALSE Otherwise.
2209  */
2210 BOOL WINAPI StrIsIntlEqualA(BOOL bCase, LPCSTR lpszStr, LPCSTR lpszComp,
2211                             int iLen)
2212 {
2213   DWORD dwFlags;
2214 
2215   TRACE("(%d,%s,%s,%d)\n", bCase,
2216         debugstr_a(lpszStr), debugstr_a(lpszComp), iLen);
2217 
2218   /* FIXME: This flag is undocumented and unknown by our CompareString.
2219    *        We need a define for it.
2220    */
2221   dwFlags = 0x10000000;
2222   if (!bCase) dwFlags |= NORM_IGNORECASE;
2223 
2224   return (CompareStringA(GetThreadLocale(), dwFlags, lpszStr, iLen, lpszComp, iLen) == CSTR_EQUAL);
2225 }
2226 
2227 /*************************************************************************
2228  * StrIsIntlEqualW	[SHLWAPI.@]
2229  *
2230  * See StrIsIntlEqualA.
2231  */
2232 BOOL WINAPI StrIsIntlEqualW(BOOL bCase, LPCWSTR lpszStr, LPCWSTR lpszComp,
2233                             int iLen)
2234 {
2235   DWORD dwFlags;
2236 
2237   TRACE("(%d,%s,%s,%d)\n", bCase,
2238         debugstr_w(lpszStr),debugstr_w(lpszComp), iLen);
2239 
2240   /* FIXME: This flag is undocumented and unknown by our CompareString.
2241    *        We need a define for it.
2242    */
2243   dwFlags = 0x10000000;
2244   if (!bCase) dwFlags |= NORM_IGNORECASE;
2245 
2246   return (CompareStringW(GetThreadLocale(), dwFlags, lpszStr, iLen, lpszComp, iLen) == CSTR_EQUAL);
2247 }
2248 
2249 /*************************************************************************
2250  * @    [SHLWAPI.399]
2251  *
2252  * Copy a string to another string, up to a maximum number of characters.
2253  *
2254  * PARAMS
2255  *  lpszDest [O] Destination string
2256  *  lpszSrc  [I] Source string
2257  *  iLen     [I] Maximum number of chars to copy
2258  *
2259  * RETURNS
2260  *  Success: A pointer to the last character written to lpszDest.
2261  *  Failure: lpszDest, if any arguments are invalid.
2262  */
2263 LPSTR WINAPI StrCpyNXA(LPSTR lpszDest, LPCSTR lpszSrc, int iLen)
2264 {
2265   TRACE("(%p,%s,%i)\n", lpszDest, debugstr_a(lpszSrc), iLen);
2266 
2267   if (lpszDest && lpszSrc && iLen > 0)
2268   {
2269     while ((iLen-- > 1) && *lpszSrc)
2270       *lpszDest++ = *lpszSrc++;
2271     if (iLen >= 0)
2272      *lpszDest = '\0';
2273   }
2274   return lpszDest;
2275 }
2276 
2277 /*************************************************************************
2278  * @    [SHLWAPI.400]
2279  *
2280  * Unicode version of StrCpyNXA.
2281  */
2282 LPWSTR WINAPI StrCpyNXW(LPWSTR lpszDest, LPCWSTR lpszSrc, int iLen)
2283 {
2284   TRACE("(%p,%s,%i)\n", lpszDest, debugstr_w(lpszSrc), iLen);
2285 
2286   if (lpszDest && lpszSrc && iLen > 0)
2287   {
2288     while ((iLen-- > 1) && *lpszSrc)
2289       *lpszDest++ = *lpszSrc++;
2290     if (iLen >= 0)
2291      *lpszDest = '\0';
2292   }
2293   return lpszDest;
2294 }
2295 
2296 /*************************************************************************
2297  * StrCmpLogicalW	[SHLWAPI.@]
2298  *
2299  * Compare two strings, ignoring case and comparing digits as numbers.
2300  *
2301  * PARAMS
2302  *  lpszStr  [I] First string to compare
2303  *  lpszComp [I] Second string to compare
2304  *  iLen     [I] Length to compare
2305  *
2306  * RETURNS
2307  *  TRUE  If the strings are equal.
2308  *  FALSE Otherwise.
2309  */
2310 INT WINAPI StrCmpLogicalW(LPCWSTR lpszStr, LPCWSTR lpszComp)
2311 {
2312   INT iDiff;
2313 
2314   TRACE("(%s,%s)\n", debugstr_w(lpszStr), debugstr_w(lpszComp));
2315 
2316   if (lpszStr && lpszComp)
2317   {
2318     while (*lpszStr)
2319     {
2320       if (!*lpszComp)
2321         return 1;
2322       else if (isdigitW(*lpszStr))
2323       {
2324         int iStr, iComp;
2325 
2326         if (!isdigitW(*lpszComp))
2327           return -1;
2328 
2329         /* Compare the numbers */
2330         StrToIntExW(lpszStr, 0, &iStr);
2331         StrToIntExW(lpszComp, 0, &iComp);
2332 
2333         if (iStr < iComp)
2334           return -1;
2335         else if (iStr > iComp)
2336           return 1;
2337 
2338         /* Skip */
2339         while (isdigitW(*lpszStr))
2340           lpszStr++;
2341         while (isdigitW(*lpszComp))
2342           lpszComp++;
2343       }
2344       else if (isdigitW(*lpszComp))
2345         return 1;
2346       else
2347       {
2348         iDiff = ChrCmpIW(*lpszStr,*lpszComp);
2349         if (iDiff > 0)
2350           return 1;
2351         else if (iDiff < 0)
2352           return -1;
2353 
2354         lpszStr++;
2355         lpszComp++;
2356       }
2357     }
2358     if (*lpszComp)
2359       return -1;
2360   }
2361   return 0;
2362 }
2363 
2364 /* Structure for formatting byte strings */
2365 typedef struct tagSHLWAPI_BYTEFORMATS
2366 {
2367   LONGLONG dLimit;
2368   double   dDivisor;
2369   double   dNormaliser;
2370   int      nDecimals;
2371 #ifdef __REACTOS__
2372   UINT     nFormatID;
2373 #else
2374   WCHAR     wPrefix;
2375 #endif
2376 } SHLWAPI_BYTEFORMATS;
2377 
2378 /*************************************************************************
2379  * StrFormatByteSizeW	[SHLWAPI.@]
2380  *
2381  * Create a string containing an abbreviated byte count of up to 2^63-1.
2382  *
2383  * PARAMS
2384  *  llBytes  [I] Byte size to format
2385  *  lpszDest [I] Destination for formatted string
2386  *  cchMax   [I] Size of lpszDest
2387  *
2388  * RETURNS
2389  *  lpszDest.
2390  *
2391  * NOTES
2392  *  There is no StrFormatByteSize64W function, it is called StrFormatByteSizeW().
2393  */
2394 LPWSTR WINAPI StrFormatByteSizeW(LONGLONG llBytes, LPWSTR lpszDest, UINT cchMax)
2395 {
2396 #define KB ((ULONGLONG)1024)
2397 #define MB (KB*KB)
2398 #define GB (KB*KB*KB)
2399 #define TB (KB*KB*KB*KB)
2400 #define PB (KB*KB*KB*KB*KB)
2401 
2402   static const SHLWAPI_BYTEFORMATS bfFormats[] =
2403   {
2404 #ifdef __REACTOS__
2405     { 10*KB, 10.24, 100.0, 2, IDS_KB_FORMAT }, /* 10 KB */
2406     { 100*KB, 102.4, 10.0, 1, IDS_KB_FORMAT }, /* 100 KB */
2407     { 1000*KB, 1024.0, 1.0, 0, IDS_KB_FORMAT }, /* 1000 KB */
2408     { 10*MB, 10485.76, 100.0, 2, IDS_MB_FORMAT }, /* 10 MB */
2409     { 100*MB, 104857.6, 10.0, 1, IDS_MB_FORMAT }, /* 100 MB */
2410     { 1000*MB, 1048576.0, 1.0, 0, IDS_MB_FORMAT }, /* 1000 MB */
2411     { 10*GB, 10737418.24, 100.0, 2, IDS_GB_FORMAT }, /* 10 GB */
2412     { 100*GB, 107374182.4, 10.0, 1, IDS_GB_FORMAT }, /* 100 GB */
2413     { 1000*GB, 1073741824.0, 1.0, 0, IDS_GB_FORMAT }, /* 1000 GB */
2414     { 10*TB, 10485.76, 100.0, 2, IDS_TB_FORMAT }, /* 10 TB */
2415     { 100*TB, 104857.6, 10.0, 1, IDS_TB_FORMAT }, /* 100 TB */
2416     { 1000*TB, 1048576.0, 1.0, 0, IDS_TB_FORMAT }, /* 1000 TB */
2417     { 10*PB, 10737418.24, 100.00, 2, IDS_PB_FORMAT }, /* 10 PB */
2418     { 100*PB, 107374182.4, 10.00, 1, IDS_PB_FORMAT }, /* 100 PB */
2419     { 1000*PB, 1073741824.0, 1.00, 0, IDS_PB_FORMAT }, /* 1000 PB */
2420     { 0, 10995116277.76, 100.00, 2, IDS_EB_FORMAT } /* EB's, catch all */
2421 #else
2422     { 10*KB, 10.24, 100.0, 2, 'K' }, /* 10 KB */
2423     { 100*KB, 102.4, 10.0, 1, 'K' }, /* 100 KB */
2424     { 1000*KB, 1024.0, 1.0, 0, 'K' }, /* 1000 KB */
2425     { 10*MB, 10485.76, 100.0, 2, 'M' }, /* 10 MB */
2426     { 100*MB, 104857.6, 10.0, 1, 'M' }, /* 100 MB */
2427     { 1000*MB, 1048576.0, 1.0, 0, 'M' }, /* 1000 MB */
2428     { 10*GB, 10737418.24, 100.0, 2, 'G' }, /* 10 GB */
2429     { 100*GB, 107374182.4, 10.0, 1, 'G' }, /* 100 GB */
2430     { 1000*GB, 1073741824.0, 1.0, 0, 'G' }, /* 1000 GB */
2431     { 10*TB, 10485.76, 100.0, 2, 'T' }, /* 10 TB */
2432     { 100*TB, 104857.6, 10.0, 1, 'T' }, /* 100 TB */
2433     { 1000*TB, 1048576.0, 1.0, 0, 'T' }, /* 1000 TB */
2434     { 10*PB, 10737418.24, 100.00, 2, 'P' }, /* 10 PB */
2435     { 100*PB, 107374182.4, 10.00, 1, 'P' }, /* 100 PB */
2436     { 1000*PB, 1073741824.0, 1.00, 0, 'P' }, /* 1000 PB */
2437     { 0, 10995116277.76, 100.00, 2, 'E' } /* EB's, catch all */
2438 #endif
2439   };
2440 #ifdef __REACTOS__
2441   WCHAR szBuff[40], wszFormat[40];
2442 #else
2443   WCHAR wszAdd[] = {' ','?','B',0};
2444 #endif
2445   double dBytes;
2446   UINT i = 0;
2447 
2448   TRACE("(0x%s,%p,%d)\n", wine_dbgstr_longlong(llBytes), lpszDest, cchMax);
2449 
2450   if (!lpszDest || !cchMax)
2451     return lpszDest;
2452 
2453   if (llBytes < 1024)  /* 1K */
2454   {
2455     WCHAR wszBytesFormat[64];
2456     LoadStringW(shlwapi_hInstance, IDS_BYTES_FORMAT, wszBytesFormat, 64);
2457     snprintfW(lpszDest, cchMax, wszBytesFormat, (int)llBytes);
2458     return lpszDest;
2459   }
2460 
2461   /* Note that if this loop completes without finding a match, i will be
2462    * pointing at the last entry, which is a catch all for > 1000 PB
2463    */
2464   while (i < sizeof(bfFormats) / sizeof(SHLWAPI_BYTEFORMATS) - 1)
2465   {
2466     if (llBytes < bfFormats[i].dLimit)
2467       break;
2468     i++;
2469   }
2470   /* Above 1 TB we encounter problems with FP accuracy. So for amounts above
2471    * this number we integer shift down by 1 MB first. The table above has
2472    * the divisors scaled down from the '< 10 TB' entry onwards, to account
2473    * for this. We also add a small fudge factor to get the correct result for
2474    * counts that lie exactly on a 1024 byte boundary.
2475    */
2476   if (i > 8)
2477     dBytes = (double)(llBytes >> 20) + 0.001; /* Scale down by 1 MB */
2478   else
2479     dBytes = (double)llBytes + 0.00001;
2480 
2481   dBytes = floor(dBytes / bfFormats[i].dDivisor) / bfFormats[i].dNormaliser;
2482 
2483 #ifdef __REACTOS__
2484   if (!FormatDouble(dBytes, bfFormats[i].nDecimals, szBuff, ARRAYSIZE(szBuff)))
2485     return NULL;
2486   LoadStringW(shlwapi_hInstance, bfFormats[i].nFormatID, wszFormat, ARRAYSIZE(wszFormat));
2487   snprintfW(lpszDest, cchMax, wszFormat, szBuff);
2488 #else
2489   if (!FormatDouble(dBytes, bfFormats[i].nDecimals, lpszDest, cchMax))
2490     return NULL;
2491   wszAdd[1] = bfFormats[i].wPrefix;
2492   StrCatBuffW(lpszDest, wszAdd, cchMax);
2493 #endif
2494   return lpszDest;
2495 }
2496 
2497 /*************************************************************************
2498  * StrFormatByteSize64A	[SHLWAPI.@]
2499  *
2500  * See StrFormatByteSizeW.
2501  */
2502 LPSTR WINAPI StrFormatByteSize64A(LONGLONG llBytes, LPSTR lpszDest, UINT cchMax)
2503 {
2504   WCHAR wszBuff[32];
2505 
2506   StrFormatByteSizeW(llBytes, wszBuff, sizeof(wszBuff)/sizeof(WCHAR));
2507 
2508   if (lpszDest)
2509     WideCharToMultiByte(CP_ACP, 0, wszBuff, -1, lpszDest, cchMax, 0, 0);
2510   return lpszDest;
2511 }
2512 
2513 /*************************************************************************
2514  * StrFormatByteSizeA	[SHLWAPI.@]
2515  *
2516  * Create a string containing an abbreviated byte count of up to 2^31-1.
2517  *
2518  * PARAMS
2519  *  dwBytes  [I] Byte size to format
2520  *  lpszDest [I] Destination for formatted string
2521  *  cchMax   [I] Size of lpszDest
2522  *
2523  * RETURNS
2524  *  lpszDest.
2525  *
2526  * NOTES
2527  *  The Ascii and Unicode versions of this function accept a different
2528  *  integer type for dwBytes. See StrFormatByteSize64A().
2529  */
2530 LPSTR WINAPI StrFormatByteSizeA(DWORD dwBytes, LPSTR lpszDest, UINT cchMax)
2531 {
2532   TRACE("(%d,%p,%d)\n", dwBytes, lpszDest, cchMax);
2533 
2534   return StrFormatByteSize64A(dwBytes, lpszDest, cchMax);
2535 }
2536 
2537 /*************************************************************************
2538  *      @	[SHLWAPI.162]
2539  *
2540  * Remove a hanging lead byte from the end of a string, if present.
2541  *
2542  * PARAMS
2543  *  lpStr [I] String to check for a hanging lead byte
2544  *  size  [I] Length of lpStr
2545  *
2546  * RETURNS
2547  *  Success: The new length of the string. Any hanging lead bytes are removed.
2548  *  Failure: 0, if any parameters are invalid.
2549  */
2550 DWORD WINAPI SHTruncateString(LPSTR lpStr, DWORD size)
2551 {
2552   if (lpStr && size)
2553   {
2554     LPSTR lastByte = lpStr + size - 1;
2555 
2556     while(lpStr < lastByte)
2557       lpStr += IsDBCSLeadByte(*lpStr) ? 2 : 1;
2558 
2559     if(lpStr == lastByte && IsDBCSLeadByte(*lpStr))
2560     {
2561       *lpStr = '\0';
2562       size--;
2563     }
2564     return size;
2565   }
2566   return 0;
2567 }
2568 
2569 /*************************************************************************
2570  *      @	[SHLWAPI.203]
2571  *
2572  * Remove a single non-trailing ampersand ('&') from a string.
2573  *
2574  * PARAMS
2575  *  lpszStr [I/O] String to remove ampersand from.
2576  *
2577  * RETURNS
2578  *  The character after the first ampersand in lpszStr, or the first character
2579  *  in lpszStr if there is no ampersand in the string.
2580  */
2581 char WINAPI SHStripMneumonicA(LPCSTR lpszStr)
2582 {
2583   LPSTR lpszIter, lpszTmp;
2584   char ch;
2585 
2586   TRACE("(%s)\n", debugstr_a(lpszStr));
2587 
2588   ch = *lpszStr;
2589 
2590   if ((lpszIter = StrChrA(lpszStr, '&')))
2591   {
2592     lpszTmp = CharNextA(lpszIter);
2593     if (*lpszTmp)
2594     {
2595       if (*lpszTmp != '&')
2596         ch =  *lpszTmp;
2597 
2598       memmove( lpszIter, lpszTmp, strlen(lpszTmp) + 1 );
2599     }
2600   }
2601 
2602   return ch;
2603 }
2604 
2605 /*************************************************************************
2606  *      @	[SHLWAPI.225]
2607  *
2608  * Unicode version of SHStripMneumonicA.
2609  */
2610 WCHAR WINAPI SHStripMneumonicW(LPCWSTR lpszStr)
2611 {
2612   LPWSTR lpszIter, lpszTmp;
2613   WCHAR ch;
2614 
2615   TRACE("(%s)\n", debugstr_w(lpszStr));
2616 
2617   ch = *lpszStr;
2618 
2619   if ((lpszIter = StrChrW(lpszStr, '&')))
2620   {
2621     lpszTmp = lpszIter + 1;
2622     if (*lpszTmp)
2623     {
2624       if (*lpszTmp != '&')
2625         ch =  *lpszTmp;
2626 
2627       memmove( lpszIter, lpszTmp, (strlenW(lpszTmp) + 1) * sizeof(WCHAR) );
2628     }
2629   }
2630 
2631   return ch;
2632 }
2633 
2634 /*************************************************************************
2635  *      @	[SHLWAPI.216]
2636  *
2637  * Convert an Ascii string to Unicode.
2638  *
2639  * PARAMS
2640  * dwCp     [I] Code page for the conversion
2641  * lpSrcStr [I] Source Ascii string to convert
2642  * lpDstStr [O] Destination for converted Unicode string
2643  * iLen     [I] Length of lpDstStr
2644  *
2645  * RETURNS
2646  *  The return value of the MultiByteToWideChar() function called on lpSrcStr.
2647  */
2648 DWORD WINAPI SHAnsiToUnicodeCP(DWORD dwCp, LPCSTR lpSrcStr, LPWSTR lpDstStr, int iLen)
2649 {
2650   DWORD dwRet;
2651 
2652   dwRet = MultiByteToWideChar(dwCp, 0, lpSrcStr, -1, lpDstStr, iLen);
2653   TRACE("%s->%s,ret=%d\n", debugstr_a(lpSrcStr), debugstr_w(lpDstStr), dwRet);
2654   return dwRet;
2655 }
2656 
2657 /*************************************************************************
2658  *      @	[SHLWAPI.215]
2659  *
2660  * Convert an Ascii string to Unicode.
2661  *
2662  * PARAMS
2663  * lpSrcStr [I] Source Ascii string to convert
2664  * lpDstStr [O] Destination for converted Unicode string
2665  * iLen     [I] Length of lpDstStr
2666  *
2667  * RETURNS
2668  *  The return value of the MultiByteToWideChar() function called on lpSrcStr.
2669  *
2670  * NOTES
2671  *  This function simply calls SHAnsiToUnicodeCP with code page CP_ACP.
2672  */
2673 DWORD WINAPI SHAnsiToUnicode(LPCSTR lpSrcStr, LPWSTR lpDstStr, int iLen)
2674 {
2675   return SHAnsiToUnicodeCP(CP_ACP, lpSrcStr, lpDstStr, iLen);
2676 }
2677 
2678 /*************************************************************************
2679  *      @	[SHLWAPI.218]
2680  *
2681  * Convert a Unicode string to Ascii.
2682  *
2683  * PARAMS
2684  *  CodePage [I] Code page to use for the conversion
2685  *  lpSrcStr [I] Source Unicode string to convert
2686  *  lpDstStr [O] Destination for converted Ascii string
2687  *  dstlen   [I] Length of buffer at lpDstStr
2688  *
2689  * RETURNS
2690  *  Success: The length in bytes of the result at lpDstStr (including the terminator)
2691  *  Failure: When using CP_UTF8, CP_UTF7 or 0xc350 as codePage, 0 is returned and
2692  *           the result is not nul-terminated.
2693  *           When using a different codepage, the length in bytes of the truncated
2694  *           result at lpDstStr (including the terminator) is returned and
2695  *           lpDstStr is always nul-terminated.
2696  *
2697  */
2698 DWORD WINAPI SHUnicodeToAnsiCP(UINT CodePage, LPCWSTR lpSrcStr, LPSTR lpDstStr, int dstlen)
2699 {
2700   static const WCHAR emptyW[] = { '\0' };
2701   int len , reqLen;
2702   LPSTR mem;
2703 
2704   if (!lpDstStr || !dstlen)
2705     return 0;
2706 
2707   if (!lpSrcStr)
2708     lpSrcStr = emptyW;
2709 
2710   *lpDstStr = '\0';
2711 
2712   len = strlenW(lpSrcStr) + 1;
2713 
2714   switch (CodePage)
2715   {
2716   case CP_WINUNICODE:
2717     CodePage = CP_UTF8; /* Fall through... */
2718   case 0x0000C350: /* FIXME: CP_ #define */
2719   case CP_UTF7:
2720   case CP_UTF8:
2721     {
2722       DWORD dwMode = 0;
2723       INT lenW = len - 1;
2724       INT needed = dstlen - 1;
2725       HRESULT hr;
2726 
2727       /* try the user supplied buffer first */
2728       hr = ConvertINetUnicodeToMultiByte(&dwMode, CodePage, lpSrcStr, &lenW, lpDstStr, &needed);
2729       if (hr == S_OK)
2730       {
2731         lpDstStr[needed] = '\0';
2732         return needed + 1;
2733       }
2734 
2735       /* user buffer too small. exclude termination and copy as much as possible */
2736       lenW = len;
2737       hr = ConvertINetUnicodeToMultiByte(&dwMode, CodePage, lpSrcStr, &lenW, NULL, &needed);
2738       needed++;
2739       mem = HeapAlloc(GetProcessHeap(), 0, needed);
2740       if (!mem)
2741         return 0;
2742 
2743       hr = ConvertINetUnicodeToMultiByte(&dwMode, CodePage, lpSrcStr, &len, mem, &needed);
2744       if (hr == S_OK)
2745       {
2746           reqLen = SHTruncateString(mem, dstlen);
2747           if (reqLen > 0) memcpy(lpDstStr, mem, reqLen-1);
2748       }
2749       HeapFree(GetProcessHeap(), 0, mem);
2750       return 0;
2751     }
2752   default:
2753     break;
2754   }
2755 
2756   /* try the user supplied buffer first */
2757   reqLen = WideCharToMultiByte(CodePage, 0, lpSrcStr, len, lpDstStr, dstlen, NULL, NULL);
2758 
2759   if (!reqLen && GetLastError() == ERROR_INSUFFICIENT_BUFFER)
2760   {
2761     reqLen = WideCharToMultiByte(CodePage, 0, lpSrcStr, len, NULL, 0, NULL, NULL);
2762     if (reqLen)
2763     {
2764       mem = HeapAlloc(GetProcessHeap(), 0, reqLen);
2765       if (mem)
2766       {
2767         WideCharToMultiByte(CodePage, 0, lpSrcStr, len, mem, reqLen, NULL, NULL);
2768 
2769         reqLen = SHTruncateString(mem, dstlen -1);
2770         reqLen++;
2771 
2772         lstrcpynA(lpDstStr, mem, reqLen);
2773         HeapFree(GetProcessHeap(), 0, mem);
2774         lpDstStr[reqLen-1] = '\0';
2775       }
2776     }
2777   }
2778   return reqLen;
2779 }
2780 
2781 /*************************************************************************
2782  *      @	[SHLWAPI.217]
2783  *
2784  * Convert a Unicode string to Ascii.
2785  *
2786  * PARAMS
2787  *  lpSrcStr [I] Source Unicode string to convert
2788  *  lpDstStr [O] Destination for converted Ascii string
2789  *  iLen     [O] Length of lpDstStr in characters
2790  *
2791  * RETURNS
2792  *  See SHUnicodeToAnsiCP
2793 
2794  * NOTES
2795  *  This function simply calls SHUnicodeToAnsiCP() with CodePage = CP_ACP.
2796  */
2797 INT WINAPI SHUnicodeToAnsi(LPCWSTR lpSrcStr, LPSTR lpDstStr, INT iLen)
2798 {
2799     return SHUnicodeToAnsiCP(CP_ACP, lpSrcStr, lpDstStr, iLen);
2800 }
2801 
2802 /*************************************************************************
2803  *      @	[SHLWAPI.345]
2804  *
2805  * Copy one string to another.
2806  *
2807  * PARAMS
2808  *  lpszSrc [I] Source string to copy
2809  *  lpszDst [O] Destination for copy
2810  *  iLen    [I] Length of lpszDst in characters
2811  *
2812  * RETURNS
2813  *  The length of the copied string, including the terminating NUL. lpszDst
2814  *  contains iLen characters of lpszSrc.
2815  */
2816 DWORD WINAPI SHAnsiToAnsi(LPCSTR lpszSrc, LPSTR lpszDst, int iLen)
2817 {
2818     LPSTR lpszRet;
2819 
2820     TRACE("(%s,%p,0x%08x)\n", debugstr_a(lpszSrc), lpszDst, iLen);
2821 
2822     lpszRet = StrCpyNXA(lpszDst, lpszSrc, iLen);
2823     return lpszRet - lpszDst + 1;
2824 }
2825 
2826 /*************************************************************************
2827  *      @	[SHLWAPI.346]
2828  *
2829  * Unicode version of SSHAnsiToAnsi.
2830  */
2831 DWORD WINAPI SHUnicodeToUnicode(LPCWSTR lpszSrc, LPWSTR lpszDst, int iLen)
2832 {
2833     LPWSTR lpszRet;
2834 
2835     TRACE("(%s,%p,0x%08x)\n", debugstr_w(lpszSrc), lpszDst, iLen);
2836 
2837     lpszRet = StrCpyNXW(lpszDst, lpszSrc, iLen);
2838     return lpszRet - lpszDst + 1;
2839 }
2840 
2841 /*************************************************************************
2842  *      @	[SHLWAPI.364]
2843  *
2844  * Determine if an Ascii string converts to Unicode and back identically.
2845  *
2846  * PARAMS
2847  *  lpSrcStr [I] Source Unicode string to convert
2848  *  lpDst    [O] Destination for resulting Ascii string
2849  *  iLen     [I] Length of lpDst in characters
2850  *
2851  * RETURNS
2852  *  TRUE, since Ascii strings always convert identically.
2853  */
2854 BOOL WINAPI DoesStringRoundTripA(LPCSTR lpSrcStr, LPSTR lpDst, INT iLen)
2855 {
2856   lstrcpynA(lpDst, lpSrcStr, iLen);
2857   return TRUE;
2858 }
2859 
2860 /*************************************************************************
2861  *      @	[SHLWAPI.365]
2862  *
2863  * Determine if a Unicode string converts to Ascii and back identically.
2864  *
2865  * PARAMS
2866  *  lpSrcStr [I] Source Unicode string to convert
2867  *  lpDst    [O] Destination for resulting Ascii string
2868  *  iLen     [I] Length of lpDst in characters
2869  *
2870  * RETURNS
2871  *  TRUE, if lpSrcStr converts to Ascii and back identically,
2872  *  FALSE otherwise.
2873  */
2874 BOOL WINAPI DoesStringRoundTripW(LPCWSTR lpSrcStr, LPSTR lpDst, INT iLen)
2875 {
2876     WCHAR szBuff[MAX_PATH];
2877 
2878     SHUnicodeToAnsi(lpSrcStr, lpDst, iLen);
2879     SHAnsiToUnicode(lpDst, szBuff, MAX_PATH);
2880     return !strcmpW(lpSrcStr, szBuff);
2881 }
2882 
2883 /*************************************************************************
2884  *      SHLoadIndirectString    [SHLWAPI.@]
2885  *
2886  * If passed a string that begins with '@', extract the string from the
2887  * appropriate resource, otherwise do a straight copy.
2888  *
2889  */
2890 HRESULT WINAPI SHLoadIndirectString(LPCWSTR src, LPWSTR dst, UINT dst_len, void **reserved)
2891 {
2892     WCHAR *dllname = NULL;
2893     HMODULE hmod = NULL;
2894     HRESULT hr = E_FAIL;
2895 #ifdef __REACTOS__
2896     WCHAR szExpanded[512];
2897 #endif
2898 
2899     TRACE("(%s %p %08x %p)\n", debugstr_w(src), dst, dst_len, reserved);
2900 
2901     if(src[0] == '@')
2902     {
2903         WCHAR *index_str;
2904         int index;
2905 
2906 #ifdef __REACTOS__
2907         if (wcschr(src, '%') != NULL)
2908         {
2909             ExpandEnvironmentStringsW(src, szExpanded, ARRAY_SIZE(szExpanded));
2910             src = szExpanded;
2911         }
2912 #endif
2913         dst[0] = 0;
2914         dllname = StrDupW(src + 1);
2915         index_str = strchrW(dllname, ',');
2916 
2917         if(!index_str) goto end;
2918 
2919         *index_str = 0;
2920         index_str++;
2921         index = atoiW(index_str);
2922 
2923 #ifdef __REACTOS__
2924         hmod = LoadLibraryExW(dllname, NULL, LOAD_LIBRARY_AS_DATAFILE);
2925 #else
2926         hmod = LoadLibraryW(dllname);
2927 #endif
2928         if(!hmod) goto end;
2929 
2930         if(index < 0)
2931         {
2932             if(LoadStringW(hmod, -index, dst, dst_len))
2933                 hr = S_OK;
2934         }
2935         else
2936             FIXME("can't handle non-negative indices (%d)\n", index);
2937     }
2938     else
2939     {
2940         if(dst != src)
2941             lstrcpynW(dst, src, dst_len);
2942         hr = S_OK;
2943     }
2944 
2945     TRACE("returning %s\n", debugstr_w(dst));
2946 end:
2947     if(hmod) FreeLibrary(hmod);
2948     LocalFree(dllname);
2949     return hr;
2950 }
2951 
2952 BOOL WINAPI IsCharSpaceA(CHAR c)
2953 {
2954     WORD CharType;
2955     return GetStringTypeA(GetSystemDefaultLCID(), CT_CTYPE1, &c, 1, &CharType) && (CharType & C1_SPACE);
2956 }
2957 
2958 /*************************************************************************
2959  *      @	[SHLWAPI.29]
2960  *
2961  * Determine if a Unicode character is a space.
2962  *
2963  * PARAMS
2964  *  wc [I] Character to check.
2965  *
2966  * RETURNS
2967  *  TRUE, if wc is a space,
2968  *  FALSE otherwise.
2969  */
2970 BOOL WINAPI IsCharSpaceW(WCHAR wc)
2971 {
2972     WORD CharType;
2973 
2974     return GetStringTypeW(CT_CTYPE1, &wc, 1, &CharType) && (CharType & C1_SPACE);
2975 }
2976