com_olechar.c 4.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160
  1. /*
  2. +----------------------------------------------------------------------+
  3. | PHP Version 7 |
  4. +----------------------------------------------------------------------+
  5. | Copyright (c) 1997-2018 The PHP Group |
  6. +----------------------------------------------------------------------+
  7. | This source file is subject to version 3.01 of the PHP license, |
  8. | that is bundled with this package in the file LICENSE, and is |
  9. | available through the world-wide-web at the following url: |
  10. | http://www.php.net/license/3_01.txt |
  11. | If you did not receive a copy of the PHP license and are unable to |
  12. | obtain it through the world-wide-web, please send a note to |
  13. | license@php.net so we can mail you a copy immediately. |
  14. +----------------------------------------------------------------------+
  15. | Author: Wez Furlong <wez@thebrainroom.com> |
  16. | Harald Radi <h.radi@nme.at> |
  17. +----------------------------------------------------------------------+
  18. */
  19. #ifdef HAVE_CONFIG_H
  20. #include "config.h"
  21. #endif
  22. #include "php.h"
  23. #include "php_ini.h"
  24. #include "ext/standard/info.h"
  25. #include "php_com_dotnet.h"
  26. #include "php_com_dotnet_internal.h"
  27. PHP_COM_DOTNET_API OLECHAR *php_com_string_to_olestring(char *string, size_t string_len, int codepage)
  28. {
  29. OLECHAR *olestring = NULL;
  30. DWORD flags = codepage == CP_UTF8 ? 0 : MB_PRECOMPOSED | MB_ERR_INVALID_CHARS;
  31. BOOL ok;
  32. if (string_len == -1) {
  33. /* determine required length for the buffer (includes NUL terminator) */
  34. string_len = MultiByteToWideChar(codepage, flags, string, -1, NULL, 0);
  35. } else {
  36. /* allow room for NUL terminator */
  37. string_len++;
  38. }
  39. if (string_len > 0) {
  40. olestring = (OLECHAR*)safe_emalloc(string_len, sizeof(OLECHAR), 0);
  41. /* XXX if that's a real multibyte string, olestring is obviously allocated excessively.
  42. This should be fixed by reallocating the olestring, but as emalloc is used, that doesn't
  43. matter much. */
  44. ok = MultiByteToWideChar(codepage, flags, string, (int)string_len, olestring, (int)string_len);
  45. if (ok > 0 && (size_t)ok < string_len) {
  46. olestring[ok] = '\0';
  47. }
  48. } else {
  49. ok = FALSE;
  50. olestring = (OLECHAR*)emalloc(sizeof(OLECHAR));
  51. *olestring = 0;
  52. }
  53. if (!ok) {
  54. char *msg = php_win32_error_to_msg(GetLastError());
  55. php_error_docref(NULL, E_WARNING,
  56. "Could not convert string to unicode: `%s'", msg);
  57. LocalFree(msg);
  58. }
  59. return olestring;
  60. }
  61. PHP_COM_DOTNET_API char *php_com_olestring_to_string(OLECHAR *olestring, size_t *string_len, int codepage)
  62. {
  63. char *string;
  64. uint32_t length = 0;
  65. BOOL ok;
  66. length = WideCharToMultiByte(codepage, 0, olestring, -1, NULL, 0, NULL, NULL);
  67. if (length) {
  68. string = (char*)safe_emalloc(length, sizeof(char), 0);
  69. length = WideCharToMultiByte(codepage, 0, olestring, -1, string, length, NULL, NULL);
  70. ok = length > 0;
  71. } else {
  72. string = (char*)emalloc(sizeof(char));
  73. *string = '\0';
  74. ok = FALSE;
  75. length = 0;
  76. }
  77. if (!ok) {
  78. char *msg = php_win32_error_to_msg(GetLastError());
  79. php_error_docref(NULL, E_WARNING,
  80. "Could not convert string from unicode: `%s'", msg);
  81. LocalFree(msg);
  82. }
  83. if (string_len) {
  84. *string_len = length-1;
  85. }
  86. return string;
  87. }
  88. BSTR php_com_string_to_bstr(zend_string *string, int codepage)
  89. {
  90. BSTR bstr = NULL;
  91. DWORD flags = codepage == CP_UTF8 ? 0 : MB_PRECOMPOSED | MB_ERR_INVALID_CHARS;
  92. size_t mb_len = ZSTR_LEN(string);
  93. int wc_len;
  94. if ((wc_len = MultiByteToWideChar(codepage, flags, ZSTR_VAL(string), (int)mb_len + 1, NULL, 0)) <= 0) {
  95. goto fail;
  96. }
  97. if ((bstr = SysAllocStringLen(NULL, (UINT)(wc_len - 1))) == NULL) {
  98. goto fail;
  99. }
  100. if ((wc_len = MultiByteToWideChar(codepage, flags, ZSTR_VAL(string), (int)mb_len + 1, bstr, wc_len)) <= 0) {
  101. goto fail;
  102. }
  103. return bstr;
  104. fail:
  105. char *msg = php_win32_error_to_msg(GetLastError());
  106. php_error_docref(NULL, E_WARNING,
  107. "Could not convert string to unicode: `%s'", msg);
  108. LocalFree(msg);
  109. SysFreeString(bstr);
  110. return SysAllocString(L"");
  111. }
  112. zend_string *php_com_bstr_to_string(BSTR bstr, int codepage)
  113. {
  114. zend_string *string = NULL;
  115. UINT wc_len = SysStringLen(bstr);
  116. int mb_len;
  117. mb_len = WideCharToMultiByte(codepage, 0, bstr, wc_len + 1, NULL, 0, NULL, NULL);
  118. if (mb_len > 0) {
  119. string = zend_string_alloc(mb_len - 1, 0);
  120. mb_len = WideCharToMultiByte(codepage, 0, bstr, wc_len + 1, ZSTR_VAL(string), mb_len, NULL, NULL);
  121. }
  122. if (mb_len <= 0) {
  123. char *msg = php_win32_error_to_msg(GetLastError());
  124. php_error_docref(NULL, E_WARNING,
  125. "Could not convert string from unicode: `%s'", msg);
  126. LocalFree(msg);
  127. if (string != NULL) {
  128. zend_string_release(string);
  129. }
  130. string = ZSTR_EMPTY_ALLOC();
  131. }
  132. return string;
  133. }