421 lines
11 KiB
C
421 lines
11 KiB
C
|
/*++
|
||
|
|
||
|
Copyright (c) 1993 Microsoft Corporation
|
||
|
|
||
|
Module Name:
|
||
|
|
||
|
spstring.c
|
||
|
|
||
|
Abstract:
|
||
|
|
||
|
This module contains functions to manipulate strings.
|
||
|
These functions would ordinarily be performed by C Runtime routines
|
||
|
except that we want to avoid linking this device driver with
|
||
|
the kernel crt.
|
||
|
|
||
|
Author:
|
||
|
|
||
|
Ted Miller (tedm) 15-Jan-1994
|
||
|
|
||
|
Revision History:
|
||
|
|
||
|
--*/
|
||
|
|
||
|
|
||
|
#include "spprecmp.h"
|
||
|
#pragma hdrstop
|
||
|
|
||
|
VOID
|
||
|
SpStringToUpper(
|
||
|
IN PWSTR String
|
||
|
)
|
||
|
{
|
||
|
for( ; *String; String++) {
|
||
|
*String = SpToUpper(*String);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
VOID
|
||
|
SpStringToLower(
|
||
|
IN PWSTR String
|
||
|
)
|
||
|
{
|
||
|
for( ; *String; String++) {
|
||
|
*String = SpToLower(*String);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
PWCHAR
|
||
|
SpFindCharFromListInString(
|
||
|
PWSTR String,
|
||
|
PWSTR CharList
|
||
|
)
|
||
|
{
|
||
|
PWSTR wcset;
|
||
|
|
||
|
while(*String) {
|
||
|
for(wcset=CharList; *wcset; wcset++) {
|
||
|
if(*wcset == *String) {
|
||
|
return(String);
|
||
|
}
|
||
|
}
|
||
|
String++;
|
||
|
}
|
||
|
return(NULL);
|
||
|
}
|
||
|
|
||
|
|
||
|
unsigned
|
||
|
SpMultiByteStringToUnsigned(
|
||
|
IN PUCHAR String,
|
||
|
OUT PUCHAR *CharThatStoppedScan OPTIONAL
|
||
|
)
|
||
|
{
|
||
|
unsigned accum = 0;
|
||
|
|
||
|
while(*String) {
|
||
|
|
||
|
if(isdigit(*String)) {
|
||
|
accum *= 10;
|
||
|
accum += *String - '0';
|
||
|
}
|
||
|
|
||
|
String++;
|
||
|
}
|
||
|
|
||
|
if(CharThatStoppedScan) {
|
||
|
*CharThatStoppedScan = String;
|
||
|
}
|
||
|
|
||
|
return(accum);
|
||
|
}
|
||
|
|
||
|
|
||
|
LONG
|
||
|
SpStringToLong(
|
||
|
IN PWSTR String,
|
||
|
OUT PWCHAR *EndOfValue,
|
||
|
IN unsigned Radix
|
||
|
)
|
||
|
{
|
||
|
PWSTR p;
|
||
|
BOOLEAN Negative;
|
||
|
LONG Accum,v;
|
||
|
WCHAR HighestDigitAllowed,HighestLetterAllowed;
|
||
|
WCHAR c;
|
||
|
|
||
|
//
|
||
|
// Validate radix, 0 or 2-36.
|
||
|
//
|
||
|
if((Radix == 1) || (Radix > 36)) {
|
||
|
if(EndOfValue) {
|
||
|
*EndOfValue = String;
|
||
|
}
|
||
|
return(0);
|
||
|
}
|
||
|
|
||
|
p = String;
|
||
|
|
||
|
//
|
||
|
// Skip whitespace.
|
||
|
//
|
||
|
while(SpIsSpace(*p)) {
|
||
|
p++;
|
||
|
}
|
||
|
|
||
|
//
|
||
|
// First char may be a plus or minus.
|
||
|
//
|
||
|
Negative = FALSE;
|
||
|
if(*p == L'-') {
|
||
|
Negative = TRUE;
|
||
|
p++;
|
||
|
} else {
|
||
|
if(*p == L'+') {
|
||
|
p++;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if(!Radix) {
|
||
|
if(*p == L'0') {
|
||
|
//
|
||
|
// Octal number
|
||
|
//
|
||
|
Radix = 8;
|
||
|
p++;
|
||
|
if((*p == L'x') || (*p == L'X')) {
|
||
|
//
|
||
|
// hex number
|
||
|
//
|
||
|
Radix = 16;
|
||
|
p++;
|
||
|
}
|
||
|
} else {
|
||
|
Radix = 10;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
HighestDigitAllowed = (Radix < 10) ? L'0'+(WCHAR)(Radix-1) : L'9';
|
||
|
HighestLetterAllowed = (Radix > 10) ? L'A'+(WCHAR)(Radix-11) : 0;
|
||
|
|
||
|
Accum = 0;
|
||
|
|
||
|
while(1) {
|
||
|
|
||
|
c = *p;
|
||
|
|
||
|
if((c >= L'0') && (c <= HighestDigitAllowed)) {
|
||
|
v = c - L'0';
|
||
|
} else {
|
||
|
|
||
|
c = SpToUpper(c);
|
||
|
|
||
|
if((c >= L'A') && (c <= HighestLetterAllowed)) {
|
||
|
v = c - L'A' + 10;
|
||
|
} else {
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
Accum *= Radix;
|
||
|
Accum += v;
|
||
|
|
||
|
p++;
|
||
|
}
|
||
|
|
||
|
if(EndOfValue) {
|
||
|
*EndOfValue = p;
|
||
|
}
|
||
|
|
||
|
return(Negative ? (0-Accum) : Accum);
|
||
|
}
|
||
|
|
||
|
PWCHAR
|
||
|
SpConvertMultiSzStrToWstr(
|
||
|
IN PCHAR Source,
|
||
|
IN ULONG Length
|
||
|
)
|
||
|
{
|
||
|
NTSTATUS status;
|
||
|
PCHAR s, sourceEnd;
|
||
|
PWCHAR dest, d;
|
||
|
ANSI_STRING ansiString;
|
||
|
UNICODE_STRING unicodeString;
|
||
|
|
||
|
if (Length <= 2) {
|
||
|
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
#if DBG
|
||
|
for (s = Source; *s != '\0'; s += strlen(s) + 1) {
|
||
|
}
|
||
|
ASSERT(Length == (ULONG)(s - Source) + 1);
|
||
|
#endif
|
||
|
|
||
|
dest = SpMemAlloc(Length * sizeof(WCHAR));
|
||
|
if (dest) {
|
||
|
|
||
|
s = Source;
|
||
|
for (sourceEnd = s + Length, d = dest;
|
||
|
s < sourceEnd && *s != '\0';
|
||
|
s += strlen(s) + 1) {
|
||
|
|
||
|
RtlInitAnsiString(&ansiString, s);
|
||
|
status = RtlAnsiStringToUnicodeString(&unicodeString, &ansiString, TRUE);
|
||
|
if (!NT_SUCCESS(status)) {
|
||
|
|
||
|
SpMemFree(dest);
|
||
|
return NULL;
|
||
|
}
|
||
|
RtlCopyMemory(d, unicodeString.Buffer, unicodeString.Length + sizeof(UNICODE_NULL));
|
||
|
d += (unicodeString.Length + sizeof(UNICODE_NULL)) / sizeof(WCHAR);
|
||
|
RtlFreeUnicodeString(&unicodeString);
|
||
|
}
|
||
|
if (s < sourceEnd) {
|
||
|
*d = UNICODE_NULL;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return dest;
|
||
|
}
|
||
|
|
||
|
PCHAR
|
||
|
SpConvertMultiSzWstrToStr(
|
||
|
IN PWCHAR Source,
|
||
|
IN ULONG Length
|
||
|
)
|
||
|
{
|
||
|
NTSTATUS status;
|
||
|
PWCHAR s, sourceEnd;
|
||
|
PCHAR dest, d;
|
||
|
ANSI_STRING ansiString;
|
||
|
UNICODE_STRING unicodeString;
|
||
|
|
||
|
if (Length <= 2) {
|
||
|
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
#if DBG
|
||
|
for (s = Source; *s != UNICODE_NULL; s += wcslen(s) + 1) {
|
||
|
}
|
||
|
ASSERT(Length == (ULONG)(s - Source) + 1);
|
||
|
#endif
|
||
|
dest = SpMemAlloc(Length * sizeof(CHAR));
|
||
|
if (dest) {
|
||
|
|
||
|
s = Source;
|
||
|
for (sourceEnd = s + Length, d = dest;
|
||
|
s < sourceEnd && *s != UNICODE_NULL;
|
||
|
s += wcslen(s) + 1) {
|
||
|
|
||
|
RtlInitUnicodeString(&unicodeString, s);
|
||
|
status = RtlUnicodeStringToAnsiString(&ansiString, &unicodeString, TRUE);
|
||
|
if (!NT_SUCCESS(status)) {
|
||
|
|
||
|
SpMemFree(dest);
|
||
|
return NULL;
|
||
|
}
|
||
|
RtlCopyMemory(d, ansiString.Buffer, ansiString.Length + 1);
|
||
|
d += ansiString.Length + 1;
|
||
|
RtlFreeAnsiString(&ansiString);
|
||
|
}
|
||
|
if (s < sourceEnd) {
|
||
|
*d = '\0';
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return dest;
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
UCHAR _SpCharTypes[CTSIZE] = {
|
||
|
|
||
|
_SP_NONE, /* 00 (NUL) */
|
||
|
_SP_NONE, /* 01 (SOH) */
|
||
|
_SP_NONE, /* 02 (STX) */
|
||
|
_SP_NONE, /* 03 (ETX) */
|
||
|
_SP_NONE, /* 04 (EOT) */
|
||
|
_SP_NONE, /* 05 (ENQ) */
|
||
|
_SP_NONE, /* 06 (ACK) */
|
||
|
_SP_NONE, /* 07 (BEL) */
|
||
|
_SP_NONE, /* 08 (BS) */
|
||
|
_SP_SPACE, /* 09 (HT) */
|
||
|
_SP_SPACE, /* 0A (LF) */
|
||
|
_SP_SPACE, /* 0B (VT) */
|
||
|
_SP_SPACE, /* 0C (FF) */
|
||
|
_SP_SPACE, /* 0D (CR) */
|
||
|
_SP_NONE, /* 0E (SI) */
|
||
|
_SP_NONE, /* 0F (SO) */
|
||
|
_SP_NONE, /* 10 (DLE) */
|
||
|
_SP_NONE, /* 11 (DC1) */
|
||
|
_SP_NONE, /* 12 (DC2) */
|
||
|
_SP_NONE, /* 13 (DC3) */
|
||
|
_SP_NONE, /* 14 (DC4) */
|
||
|
_SP_NONE, /* 15 (NAK) */
|
||
|
_SP_NONE, /* 16 (SYN) */
|
||
|
_SP_NONE, /* 17 (ETB) */
|
||
|
_SP_NONE, /* 18 (CAN) */
|
||
|
_SP_NONE, /* 19 (EM) */
|
||
|
_SP_NONE, /* 1A (SUB) */
|
||
|
_SP_NONE, /* 1B (ESC) */
|
||
|
_SP_NONE, /* 1C (FS) */
|
||
|
_SP_NONE, /* 1D (GS) */
|
||
|
_SP_NONE, /* 1E (RS) */
|
||
|
_SP_NONE, /* 1F (US) */
|
||
|
_SP_SPACE, /* 20 SPACE */
|
||
|
_SP_NONE, /* 21 ! */
|
||
|
_SP_NONE, /* 22 " */
|
||
|
_SP_NONE, /* 23 # */
|
||
|
_SP_NONE, /* 24 $ */
|
||
|
_SP_NONE, /* 25 % */
|
||
|
_SP_NONE, /* 26 & */
|
||
|
_SP_NONE, /* 27 ' */
|
||
|
_SP_NONE, /* 28 ( */
|
||
|
_SP_NONE, /* 29 ) */
|
||
|
_SP_NONE, /* 2A * */
|
||
|
_SP_NONE, /* 2B + */
|
||
|
_SP_NONE, /* 2C , */
|
||
|
_SP_NONE, /* 2D - */
|
||
|
_SP_NONE, /* 2E . */
|
||
|
_SP_NONE, /* 2F / */
|
||
|
_SP_DIGIT + _SP_XDIGIT, /* 30 0 */
|
||
|
_SP_DIGIT + _SP_XDIGIT, /* 31 1 */
|
||
|
_SP_DIGIT + _SP_XDIGIT, /* 32 2 */
|
||
|
_SP_DIGIT + _SP_XDIGIT, /* 33 3 */
|
||
|
_SP_DIGIT + _SP_XDIGIT, /* 34 4 */
|
||
|
_SP_DIGIT + _SP_XDIGIT, /* 35 5 */
|
||
|
_SP_DIGIT + _SP_XDIGIT, /* 36 6 */
|
||
|
_SP_DIGIT + _SP_XDIGIT, /* 37 7 */
|
||
|
_SP_DIGIT + _SP_XDIGIT, /* 38 8 */
|
||
|
_SP_DIGIT + _SP_XDIGIT, /* 39 9 */
|
||
|
_SP_NONE, /* 3A : */
|
||
|
_SP_NONE, /* 3B ; */
|
||
|
_SP_NONE, /* 3C < */
|
||
|
_SP_NONE, /* 3D = */
|
||
|
_SP_NONE, /* 3E > */
|
||
|
_SP_NONE, /* 3F ? */
|
||
|
_SP_NONE, /* 40 @ */
|
||
|
_SP_UPPER + _SP_XDIGIT, /* 41 A */
|
||
|
_SP_UPPER + _SP_XDIGIT, /* 42 B */
|
||
|
_SP_UPPER + _SP_XDIGIT, /* 43 C */
|
||
|
_SP_UPPER + _SP_XDIGIT, /* 44 D */
|
||
|
_SP_UPPER + _SP_XDIGIT, /* 45 E */
|
||
|
_SP_UPPER + _SP_XDIGIT, /* 46 F */
|
||
|
_SP_UPPER, /* 47 G */
|
||
|
_SP_UPPER, /* 48 H */
|
||
|
_SP_UPPER, /* 49 I */
|
||
|
_SP_UPPER, /* 4A J */
|
||
|
_SP_UPPER, /* 4B K */
|
||
|
_SP_UPPER, /* 4C L */
|
||
|
_SP_UPPER, /* 4D M */
|
||
|
_SP_UPPER, /* 4E N */
|
||
|
_SP_UPPER, /* 4F O */
|
||
|
_SP_UPPER, /* 50 P */
|
||
|
_SP_UPPER, /* 51 Q */
|
||
|
_SP_UPPER, /* 52 R */
|
||
|
_SP_UPPER, /* 53 S */
|
||
|
_SP_UPPER, /* 54 T */
|
||
|
_SP_UPPER, /* 55 U */
|
||
|
_SP_UPPER, /* 56 V */
|
||
|
_SP_UPPER, /* 57 W */
|
||
|
_SP_UPPER, /* 58 X */
|
||
|
_SP_UPPER, /* 59 Y */
|
||
|
_SP_UPPER, /* 5A Z */
|
||
|
_SP_NONE, /* 5B [ */
|
||
|
_SP_NONE, /* 5C \ */
|
||
|
_SP_NONE, /* 5D ] */
|
||
|
_SP_NONE, /* 5E ^ */
|
||
|
_SP_NONE, /* 5F _ */
|
||
|
_SP_NONE, /* 60 ` */
|
||
|
_SP_LOWER + _SP_XDIGIT, /* 61 a */
|
||
|
_SP_LOWER + _SP_XDIGIT, /* 62 b */
|
||
|
_SP_LOWER + _SP_XDIGIT, /* 63 c */
|
||
|
_SP_LOWER + _SP_XDIGIT, /* 64 d */
|
||
|
_SP_LOWER + _SP_XDIGIT, /* 65 e */
|
||
|
_SP_LOWER + _SP_XDIGIT, /* 66 f */
|
||
|
_SP_LOWER, /* 67 g */
|
||
|
_SP_LOWER, /* 68 h */
|
||
|
_SP_LOWER, /* 69 i */
|
||
|
_SP_LOWER, /* 6A j */
|
||
|
_SP_LOWER, /* 6B k */
|
||
|
_SP_LOWER, /* 6C l */
|
||
|
_SP_LOWER, /* 6D m */
|
||
|
_SP_LOWER, /* 6E n */
|
||
|
_SP_LOWER, /* 6F o */
|
||
|
_SP_LOWER, /* 70 p */
|
||
|
_SP_LOWER, /* 71 q */
|
||
|
_SP_LOWER, /* 72 r */
|
||
|
_SP_LOWER, /* 73 s */
|
||
|
_SP_LOWER, /* 74 t */
|
||
|
_SP_LOWER, /* 75 u */
|
||
|
_SP_LOWER, /* 76 v */
|
||
|
_SP_LOWER, /* 77 w */
|
||
|
_SP_LOWER, /* 78 x */
|
||
|
_SP_LOWER, /* 79 y */
|
||
|
_SP_LOWER /* 7A z */
|
||
|
};
|