2000-06-09 07:27:21 +02:00
|
|
|
/*
|
|
|
|
* Wine internal Unicode definitions
|
|
|
|
*
|
|
|
|
* Copyright 2000 Alexandre Julliard
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef __WINE_UNICODE_H
|
|
|
|
#define __WINE_UNICODE_H
|
|
|
|
|
2000-06-12 03:16:11 +02:00
|
|
|
#include "windef.h"
|
|
|
|
|
2000-06-09 07:27:21 +02:00
|
|
|
/* code page info common to SBCS and DBCS */
|
|
|
|
struct cp_info
|
|
|
|
{
|
|
|
|
unsigned int codepage; /* codepage id */
|
|
|
|
unsigned int char_size; /* char size (1 or 2 bytes) */
|
2000-06-12 03:16:11 +02:00
|
|
|
WCHAR def_char; /* default char value (can be double-byte) */
|
|
|
|
WCHAR def_unicode_char; /* default Unicode char value */
|
2000-06-09 07:27:21 +02:00
|
|
|
const char *name; /* code page name */
|
|
|
|
};
|
|
|
|
|
|
|
|
struct sbcs_table
|
|
|
|
{
|
|
|
|
struct cp_info info;
|
2000-06-12 03:16:11 +02:00
|
|
|
const WCHAR *cp2uni; /* code page -> Unicode map */
|
2000-06-09 07:27:21 +02:00
|
|
|
const unsigned char *uni2cp_low; /* Unicode -> code page map */
|
|
|
|
const unsigned short *uni2cp_high;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct dbcs_table
|
|
|
|
{
|
|
|
|
struct cp_info info;
|
2000-06-12 03:16:11 +02:00
|
|
|
const WCHAR *cp2uni; /* code page -> Unicode map */
|
2000-06-09 07:27:21 +02:00
|
|
|
const unsigned char *cp2uni_leadbytes;
|
|
|
|
const unsigned short *uni2cp_low; /* Unicode -> code page map */
|
|
|
|
const unsigned short *uni2cp_high;
|
|
|
|
unsigned char lead_bytes[12]; /* lead bytes ranges */
|
|
|
|
};
|
|
|
|
|
|
|
|
union cptable
|
|
|
|
{
|
|
|
|
struct cp_info info;
|
|
|
|
struct sbcs_table sbcs;
|
|
|
|
struct dbcs_table dbcs;
|
|
|
|
};
|
|
|
|
|
|
|
|
extern const union cptable *cp_get_table( unsigned int codepage );
|
|
|
|
extern const union cptable *cp_enum_table( unsigned int index );
|
|
|
|
|
|
|
|
extern int cp_mbstowcs( const union cptable *table, int flags,
|
|
|
|
const char *src, int srclen,
|
2000-06-12 03:16:11 +02:00
|
|
|
WCHAR *dst, int dstlen );
|
2000-06-09 07:27:21 +02:00
|
|
|
extern int cp_wcstombs( const union cptable *table, int flags,
|
2000-06-12 03:16:11 +02:00
|
|
|
const WCHAR *src, int srclen,
|
2000-06-10 06:29:16 +02:00
|
|
|
char *dst, int dstlen, const char *defchar, int *used );
|
2000-08-11 02:44:33 +02:00
|
|
|
extern int utf8_wcstombs( const WCHAR *src, int srclen, char *dst, int dstlen );
|
|
|
|
extern int utf8_mbstowcs( int flags, const char *src, int srclen, WCHAR *dst, int dstlen );
|
2000-06-10 06:29:16 +02:00
|
|
|
|
2000-06-09 07:27:21 +02:00
|
|
|
static inline int is_dbcs_leadbyte( const union cptable *table, unsigned char ch )
|
|
|
|
{
|
|
|
|
return (table->info.char_size == 2) && (table->dbcs.cp2uni_leadbytes[ch]);
|
|
|
|
}
|
|
|
|
|
2000-06-12 03:16:11 +02:00
|
|
|
static inline WCHAR tolowerW( WCHAR ch )
|
|
|
|
{
|
2000-09-07 20:42:52 +02:00
|
|
|
extern const WCHAR casemap_lower[];
|
2000-06-12 03:16:11 +02:00
|
|
|
return ch + casemap_lower[casemap_lower[ch >> 8] + (ch & 0xff)];
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline WCHAR toupperW( WCHAR ch )
|
|
|
|
{
|
2000-09-07 20:42:52 +02:00
|
|
|
extern const WCHAR casemap_upper[];
|
2000-06-12 03:16:11 +02:00
|
|
|
return ch + casemap_upper[casemap_upper[ch >> 8] + (ch & 0xff)];
|
|
|
|
}
|
|
|
|
|
2000-08-11 22:53:40 +02:00
|
|
|
/* the character type contains the C1_* flags in the low 12 bits */
|
|
|
|
/* and the C2_* type in the high 4 bits */
|
|
|
|
static inline unsigned short get_char_typeW( WCHAR ch )
|
|
|
|
{
|
2000-09-07 20:42:52 +02:00
|
|
|
extern const unsigned short wctype_table[];
|
2000-08-11 22:53:40 +02:00
|
|
|
return wctype_table[wctype_table[ch >> 8] + (ch & 0xff)];
|
|
|
|
}
|
|
|
|
|
2000-06-12 03:16:11 +02:00
|
|
|
/* some useful string manipulation routines */
|
|
|
|
|
|
|
|
static inline unsigned int strlenW( const WCHAR *str )
|
2000-06-10 06:29:16 +02:00
|
|
|
{
|
|
|
|
#if defined(__i386__) && defined(__GNUC__)
|
|
|
|
int dummy, res;
|
2000-09-13 01:41:07 +02:00
|
|
|
__asm__ __volatile__( "cld\n\t"
|
|
|
|
"repne\n\t"
|
|
|
|
"scasw\n\t"
|
|
|
|
"notl %0"
|
|
|
|
: "=c" (res), "=&D" (dummy)
|
|
|
|
: "0" (0xffffffff), "1" (str), "a" (0) );
|
2000-06-10 06:29:16 +02:00
|
|
|
return res - 1;
|
|
|
|
#else
|
2000-06-12 03:16:11 +02:00
|
|
|
const WCHAR *s = str;
|
2000-06-10 06:29:16 +02:00
|
|
|
while (*s) s++;
|
|
|
|
return s - str;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2000-06-12 03:16:11 +02:00
|
|
|
static inline WCHAR *strcpyW( WCHAR *dst, const WCHAR *src )
|
|
|
|
{
|
|
|
|
#if defined(__i386__) && defined(__GNUC__)
|
|
|
|
int dummy1, dummy2, dummy3;
|
|
|
|
__asm__ __volatile__( "cld\n"
|
|
|
|
"1:\tlodsw\n\t"
|
|
|
|
"stosw\n\t"
|
|
|
|
"testw %%ax,%%ax\n\t"
|
|
|
|
"jne 1b"
|
|
|
|
: "=&S" (dummy1), "=&D" (dummy2), "=&a" (dummy3)
|
|
|
|
: "0" (src), "1" (dst)
|
|
|
|
: "memory" );
|
|
|
|
#else
|
|
|
|
WCHAR *p = dst;
|
|
|
|
while ((*p++ = *src++));
|
|
|
|
#endif
|
|
|
|
return dst;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline int strcmpW( const WCHAR *str1, const WCHAR *str2 )
|
|
|
|
{
|
|
|
|
while (*str1 && (*str1 == *str2)) { str1++; str2++; }
|
|
|
|
return *str1 - *str2;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline int strncmpW( const WCHAR *str1, const WCHAR *str2, int n )
|
|
|
|
{
|
2000-06-13 03:08:29 +02:00
|
|
|
if (n <= 0) return 0;
|
2000-06-12 03:16:11 +02:00
|
|
|
while ((--n > 0) && *str1 && (*str1 == *str2)) { str1++; str2++; }
|
|
|
|
return *str1 - *str2;
|
|
|
|
}
|
|
|
|
|
2000-08-14 16:41:19 +02:00
|
|
|
static inline WCHAR *strncpyW( WCHAR *str1, const WCHAR *str2, int n )
|
|
|
|
{
|
|
|
|
WCHAR *ret = str1;
|
|
|
|
while (n-- > 0) if (!(*str1++ = *str2++)) break;
|
|
|
|
while (n-- > 0) *str1++ = 0;
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2000-06-12 03:16:11 +02:00
|
|
|
static inline WCHAR *strcatW( WCHAR *dst, const WCHAR *src )
|
|
|
|
{
|
|
|
|
strcpyW( dst + strlenW(dst), src );
|
|
|
|
return dst;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline WCHAR *strchrW( const WCHAR *str, WCHAR ch )
|
|
|
|
{
|
|
|
|
for ( ; *str; str++) if (*str == ch) return (WCHAR *)str;
|
|
|
|
return NULL;
|
|
|
|
}
|
2000-08-02 01:33:37 +02:00
|
|
|
|
|
|
|
static inline WCHAR *strrchrW( const WCHAR *str, WCHAR ch )
|
|
|
|
{
|
|
|
|
WCHAR *ret = NULL;
|
|
|
|
for ( ; *str; str++) if (*str == ch) ret = (WCHAR *)str;
|
|
|
|
return ret;
|
|
|
|
}
|
2000-06-12 03:16:11 +02:00
|
|
|
|
2000-06-13 03:08:29 +02:00
|
|
|
static inline WCHAR *strlwrW( WCHAR *str )
|
2000-06-12 03:16:11 +02:00
|
|
|
{
|
2000-06-13 03:08:29 +02:00
|
|
|
WCHAR *ret = str;
|
|
|
|
while ((*str = tolowerW(*str))) str++;
|
|
|
|
return ret;
|
2000-06-12 03:16:11 +02:00
|
|
|
}
|
|
|
|
|
2000-06-13 03:08:29 +02:00
|
|
|
static inline WCHAR *struprW( WCHAR *str )
|
|
|
|
{
|
|
|
|
WCHAR *ret = str;
|
|
|
|
while ((*str = toupperW(*str))) str++;
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
extern int strcmpiW( const WCHAR *str1, const WCHAR *str2 );
|
|
|
|
extern int strncmpiW( const WCHAR *str1, const WCHAR *str2, int n );
|
|
|
|
extern WCHAR *strstrW( const WCHAR *str, const WCHAR *sub );
|
|
|
|
|
2000-06-09 07:27:21 +02:00
|
|
|
#endif /* __WINE_UNICODE_H */
|