dbcs.cpp
Upload User: zhongxx05
Upload Date: 2007-06-06
Package Size: 33641k
Code Size: 10k
Category:

Symbian

Development Platform:

C/C++

  1. /* ***** BEGIN LICENSE BLOCK ***** 
  2.  * Version: RCSL 1.0/RPSL 1.0 
  3.  *  
  4.  * Portions Copyright (c) 1995-2002 RealNetworks, Inc. All Rights Reserved. 
  5.  *      
  6.  * The contents of this file, and the files included with this file, are 
  7.  * subject to the current version of the RealNetworks Public Source License 
  8.  * Version 1.0 (the "RPSL") available at 
  9.  * http://www.helixcommunity.org/content/rpsl unless you have licensed 
  10.  * the file under the RealNetworks Community Source License Version 1.0 
  11.  * (the "RCSL") available at http://www.helixcommunity.org/content/rcsl, 
  12.  * in which case the RCSL will apply. You may also obtain the license terms 
  13.  * directly from RealNetworks.  You may not use this file except in 
  14.  * compliance with the RPSL or, if you have a valid RCSL with RealNetworks 
  15.  * applicable to this file, the RCSL.  Please see the applicable RPSL or 
  16.  * RCSL for the rights, obligations and limitations governing use of the 
  17.  * contents of the file.  
  18.  *  
  19.  * This file is part of the Helix DNA Technology. RealNetworks is the 
  20.  * developer of the Original Code and owns the copyrights in the portions 
  21.  * it created. 
  22.  *  
  23.  * This file, and the files included with this file, is distributed and made 
  24.  * available on an 'AS IS' basis, WITHOUT WARRANTY OF ANY KIND, EITHER 
  25.  * EXPRESS OR IMPLIED, AND REALNETWORKS HEREBY DISCLAIMS ALL SUCH WARRANTIES, 
  26.  * INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF MERCHANTABILITY, FITNESS 
  27.  * FOR A PARTICULAR PURPOSE, QUIET ENJOYMENT OR NON-INFRINGEMENT. 
  28.  * 
  29.  * Technology Compatibility Kit Test Suite(s) Location: 
  30.  *    http://www.helixcommunity.org/content/tck 
  31.  * 
  32.  * Contributor(s): 
  33.  *  
  34.  * ***** END LICENSE BLOCK ***** */ 
  35. #include "hxtypes.h"
  36. #include "hlxclib/string.h"
  37. #if defined( _WIN32 ) || defined( _WINDOWS )
  38. #include <windows.h> // for AnsiNext(), AnsiPrev()
  39. #include <ctype.h> // for isleadbyte()
  40. #endif
  41. #ifdef _MACINTOSH
  42. #ifndef _MAC_MACHO
  43. #include <script.h> // for CharByte()
  44. #endif
  45. // These aren't defined by the OS headers, so I define them here. 
  46. #define  MBCS_LEADBYTE      -1
  47. #define  MBCS_SINGLEBYTE    0
  48. #define  MBCS_LASTBYTE      1
  49. #define  MBCS_MIDDLEBYTE    2
  50. // Functions used to simplify the logic of moving to either end of a character.
  51. char* MoveToLeadByte(char *pChar);
  52. char* MoveToLastByte(char *pChar);
  53. #endif
  54. #include "dbcs.h"
  55. #include "hxassert.h"
  56. #include "hxheap.h"
  57. #ifdef _DEBUG
  58. #undef HX_THIS_FILE
  59. static const char HX_THIS_FILE[] = __FILE__;
  60. #endif
  61. /* optimized implementations for Unix are in dbcs.h */
  62. #if !defined(_UNIX) && !defined(_SYMBIAN)
  63. BOOL HXIsDBCSEnabled()
  64. {
  65. #if (defined( _WIN32 ) || defined( _WINDOWS )) && !defined(WIN32_PLATFORM_PSPC)
  66.     static int DBCSEnabled = -1;
  67.     if(DBCSEnabled < 0)
  68.         DBCSEnabled = GetSystemMetrics(SM_DBCSENABLED); 
  69. #elif defined(_MACINTOSH)
  70.     int DBCSEnabled = 1;
  71. #else
  72.     int DBCSEnabled = 0;
  73. #endif
  74.     return DBCSEnabled;
  75. }
  76. // returns pointer to the next character in multi-bytes CS string
  77. char*
  78. HXGetNextChar(const char* pChar)
  79. {
  80. if(!pChar)
  81. return NULL;
  82. if(!HXIsDBCSEnabled())
  83.     return (char *)(pChar + 1);
  84. char * pNext = NULL;
  85. #if (defined( _WIN32 ) || defined( _WINDOWS )) && !defined(WIN32_PLATFORM_PSPC)
  86. pNext = AnsiNext(pChar);
  87. #elif defined (_MACINTOSH)
  88. pNext=(char*)pChar;
  89. // If this is a multi-byte character then return the last byte of the character.
  90. // If it is a singbyte character then do nothing.
  91. pNext=MoveToLastByte(pNext);
  92. // If this is a single byte character then move to the next byte.
  93. // If this is a multi-byte character then we know we are at the end of the character,
  94. // so moving + 1 would correct.
  95. pNext=pNext+1;
  96. #else  //unix
  97. pNext = (char*)pChar + 1;
  98. #endif 
  99. if(pNext == pChar)
  100. return pNext + 1;
  101. return pNext;
  102. }
  103. // returns pointer to the previous character in multi-bytes CS string
  104. char*
  105. HXGetPrevChar(const char* pStart, const char* pChar)
  106. {
  107. if(!pChar)
  108. return NULL;
  109. if(!HXIsDBCSEnabled())
  110.     return (char *)(pChar - 1);
  111. char * pPrev = NULL;
  112. #if (defined( _WIN32 ) || defined( _WINDOWS )) && !defined(WIN32_PLATFORM_PSPC)
  113. pPrev = AnsiPrev(pStart, pChar);
  114. #elif defined(_MACINTOSH)
  115. // Since there is no AnsiPrev equivalent on the Macintosh. 
  116.      
  117.     pPrev=(char*)pChar;
  118.     
  119.     //Move to LeadByte of current character.
  120.     // If single byte do nothing.
  121.     pPrev=MoveToLeadByte(pPrev);
  122.     
  123.     //Move back one byte.
  124.     pPrev=pPrev-1;
  125.     
  126.     //Move to leadbyte of this character.
  127.     //If single byte do nothing. 
  128.     pPrev=MoveToLeadByte(pPrev);
  129. #else // unix
  130. pPrev = (char*)pChar - 1;
  131. #endif
  132. if(pPrev == pChar)
  133. return pPrev - 1;
  134. return pPrev;
  135. }
  136. BOOL
  137. HXIsEqual(const char* pChar, char CharToCompare)
  138. {
  139. if(!pChar)
  140. return FALSE;
  141. if(!HXIsDBCSEnabled())
  142.     return (* pChar == CharToCompare);
  143. char * pNext = HXGetNextChar(pChar);
  144. if(!pNext || pNext - pChar == 1)
  145. {
  146. if(*pChar == CharToCompare)
  147. return TRUE;
  148. }
  149. return FALSE;
  150. }
  151. char*
  152. HXFindChar(const char* pStart, char CharToFind)
  153. {
  154. if(!HXIsDBCSEnabled())
  155.     return (char*)strchr(pStart, CharToFind);
  156. const char * pCurrent = pStart;
  157. while(pCurrent && *pCurrent)
  158. {
  159. char * pNext;
  160. pNext = HXGetNextChar(pCurrent);
  161. if(!pNext || pNext - pCurrent == 1)
  162. if(*pCurrent == CharToFind)
  163.     return (char *)pCurrent;
  164. pCurrent = pNext;
  165. }
  166. return NULL;
  167. }
  168. char*
  169. HXReverseFindChar(const char* pStart, char CharToFind)
  170. {
  171. if(!HXIsDBCSEnabled())
  172. return (char*)strrchr(pStart, CharToFind);
  173. const char * pCurrent = pStart;
  174.     
  175. char * pLastFound = NULL;
  176. while(*pCurrent)
  177. {
  178. char * pNext;
  179. pNext = HXGetNextChar(pCurrent);
  180. if(!pNext || pNext - pCurrent == 1)
  181. if(*pCurrent == CharToFind)
  182. pLastFound = (char *)pCurrent;
  183. pCurrent = pNext;
  184. }
  185. return pLastFound;
  186. }
  187. char*
  188. HXFindString(const char* pStart, const char * pString)
  189. {
  190. if(!HXIsDBCSEnabled())
  191.     return (char*)strstr(pStart, pString);
  192. const char * pCurrent = pStart;
  193.     int Length = strlen(pString);
  194. while(*pCurrent)
  195. {
  196. if(!memcmp(pCurrent, pString, Length))
  197. {
  198. const char * pTest = pCurrent;
  199. while(pTest < pCurrent + Length)
  200. pTest = HXGetNextChar(pTest);
  201. if(pTest == pCurrent + Length)
  202.     return (char *)pCurrent;
  203. }
  204. pCurrent = HXGetNextChar(pCurrent);
  205. }
  206. return NULL;
  207. }
  208. int
  209. HXCompareStrings(const char* pString1, const char* pString2, size_t count)
  210. {
  211. if(!pString1 && !pString2)
  212. return 0;
  213. if(!pString1)
  214. return -1;
  215. if(!pString2)
  216. return 1;
  217. if(!HXIsDBCSEnabled())
  218. return strncmp(pString1,pString2, count);
  219. const char * pEnd1 = pString1;
  220. const char * pEnd2 = pString2;
  221. while(*pEnd1 && (size_t)(pEnd1 - pString1) < count)
  222. pEnd1 = HXGetNextChar( pEnd1 );
  223. while(*pEnd2 && (size_t)(pEnd2 - pString2) < count)
  224. pEnd2 = HXGetNextChar( pEnd2 );
  225. if(pEnd1 - pString1 < pEnd2 - pString2)
  226. return -1;
  227. if(pEnd1 - pString1 > pEnd2 - pString2)
  228. return 1;
  229. return memcmp(pString1, pString2, pEnd1 - pString1);
  230. }
  231. BOOL
  232. HXIsLeadByte(char Byte)
  233. {
  234. if(!HXIsDBCSEnabled()) return(FALSE);
  235. #if (defined( _WIN32 ) || defined( _WINDOWS )) && !defined(WIN32_PLATFORM_PSPC)
  236. return _ismbblead((unsigned int)Byte);
  237. #elif defined(_MACINTOSH) || defined(_MAC_UNIX)
  238. #if defined(_CARBON) || defined(_MAC_UNIX)
  239. return (CharacterType((Ptr)&Byte,0, smRoman) == MBCS_LEADBYTE);
  240. #else
  241. return (CharType((Ptr)&Byte,0) == MBCS_LEADBYTE);
  242. #endif
  243. #else
  244. return FALSE;
  245. #endif
  246. }
  247. // We do not want a LeadByte to be hanging loose.  If the last character
  248. // to copy is a LeadByte, replace it with a null character.
  249. char* HXCopyStringN(char* pOutput, const char* pInput, size_t count)
  250. {
  251. HX_ASSERT(count > 0);
  252. if(!HXIsDBCSEnabled() || count == 0 || strlen(pInput) <= count)
  253. return (char*)strncpy(pOutput, pInput, count); /* Flawfinder: ignore */
  254. const char * pEnd = pInput + count;
  255. const char * pPrev = HXGetPrevChar(pInput, pEnd);
  256. if (pEnd != HXGetNextChar(pPrev))
  257. {
  258. memset(pOutput, 0, count);
  259. pEnd = pPrev;
  260. }
  261. return (char*)strncpy(pOutput, pInput, pEnd - pInput); /* Flawfinder: ignore */
  262. }
  263. //
  264. // Please return 0 on a non-dbcs machine.  on a dbcs machine, return the code page number active.
  265. //
  266. INT32
  267. GetDBCSCodePage(void) 
  268. {
  269. #if defined( _WINDOWS ) && !defined( WIN32_PLATFORM_PSPC )
  270. return _getmbcp(); 
  271. #else 
  272. return 0; 
  273. #endif
  274. }
  275. //////////////////////////////
  276. // MACINTOSH ONLY FUNCTIONS
  277. //
  278. #ifdef _MACINTOSH
  279. //* Move to the first byte of the current character.
  280. //* Does nothing if the current character is a single byte.
  281. char* MoveToLeadByte(char* pChar) {
  282. char *pPrev=pChar;
  283. #if 0 /* CharByte can't be executed at interrupt time */
  284. if (CharByte((Ptr)pPrev,0) != MBCS_SINGLEBYTE) {
  285. while (CharByte((Ptr)pPrev,0) != MBCS_LEADBYTE) 
  286.        pPrev=pPrev-1;
  287. }//*if
  288. #endif
  289. return(pPrev);
  290. }
  291. //* Move to the last byte of the current character.
  292. //* Okay this is strange logic, CharByte searches upto the offset that is specified.
  293. //* It never looks back, so it must always be at the beginning of a character in order to 
  294. //* ever return a correct value, with the exception of the MBCS_LEADBYTE which could be specified
  295. //* as 1 byte.
  296. //* Does nothing if the current character is a single byte.
  297. char* MoveToLastByte(char *pChar) {
  298. char *pNext=pChar;
  299. #if 0 /* CharByte can't be executed at interrupt time */
  300. if (CharByte((Ptr)pNext,1) != MBCS_SINGLEBYTE) {
  301. if (CharByte((Ptr)pNext,1) == MBCS_LASTBYTE) 
  302.        pNext=pNext+1;
  303. }//*if
  304. #endif
  305. return(pNext);
  306. }
  307. #endif
  308. #endif // !defined(_UNIX)