mysqli_warning.c 8.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335
  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: Georg Richter <georg@php.net> |
  16. +----------------------------------------------------------------------+
  17. */
  18. #ifdef HAVE_CONFIG_H
  19. #include "config.h"
  20. #endif
  21. #include <signal.h>
  22. #include "php.h"
  23. #include "php_ini.h"
  24. #include "ext/standard/info.h"
  25. #include "php_mysqli_structs.h"
  26. #include "mysqli_priv.h"
  27. /* Define these in the PHP7 tree to make merging easy process */
  28. #define ZSTR_DUPLICATE (1<<0)
  29. #define ZSTR_AUTOFREE (1<<1)
  30. #define ZVAL_UTF8_STRING(z, s, flags) ZVAL_STRING((z), (char*)(s))
  31. #define ZVAL_UTF8_STRINGL(z, s, l, flags) ZVAL_STRINGL((z), (char*)(s), (l))
  32. /* {{{ void php_clear_warnings() */
  33. void php_clear_warnings(MYSQLI_WARNING *w)
  34. {
  35. MYSQLI_WARNING *n;
  36. while (w) {
  37. n = w;
  38. zval_ptr_dtor_str(&(w->reason));
  39. zval_ptr_dtor_str(&(w->sqlstate));
  40. w = w->next;
  41. efree(n);
  42. }
  43. }
  44. /* }}} */
  45. #ifndef MYSQLI_USE_MYSQLND
  46. /* {{{ MYSQLI_WARNING *php_new_warning */
  47. static
  48. MYSQLI_WARNING *php_new_warning(const char *reason, int errorno)
  49. {
  50. MYSQLI_WARNING *w;
  51. w = (MYSQLI_WARNING *)ecalloc(1, sizeof(MYSQLI_WARNING));
  52. ZVAL_UTF8_STRING(&(w->reason), reason, ZSTR_DUPLICATE);
  53. ZVAL_UTF8_STRINGL(&(w->sqlstate), "HY000", sizeof("HY000") - 1, ZSTR_DUPLICATE);
  54. w->errorno = errorno;
  55. return w;
  56. }
  57. /* }}} */
  58. /* {{{ MYSQLI_WARNING *php_get_warnings(MYSQL *mysql) */
  59. MYSQLI_WARNING *php_get_warnings(MYSQL *mysql)
  60. {
  61. MYSQLI_WARNING *w, *first = NULL, *prev = NULL;
  62. MYSQL_RES *result;
  63. MYSQL_ROW row;
  64. if (mysql_real_query(mysql, "SHOW WARNINGS", 13)) {
  65. return NULL;
  66. }
  67. result = mysql_store_result(mysql);
  68. while ((row = mysql_fetch_row(result))) {
  69. w = php_new_warning(row[2], atoi(row[1]));
  70. if (!first) {
  71. first = w;
  72. }
  73. if (prev) {
  74. prev->next = w;
  75. }
  76. prev = w;
  77. }
  78. mysql_free_result(result);
  79. return first;
  80. }
  81. /* }}} */
  82. #else
  83. /* {{{ MYSQLI_WARNING *php_new_warning */
  84. static
  85. MYSQLI_WARNING *php_new_warning(zval * reason, int errorno)
  86. {
  87. MYSQLI_WARNING *w;
  88. w = (MYSQLI_WARNING *)ecalloc(1, sizeof(MYSQLI_WARNING));
  89. ZVAL_COPY(&w->reason, reason);
  90. convert_to_string(&w->reason);
  91. ZVAL_UTF8_STRINGL(&(w->sqlstate), "HY000", sizeof("HY000") - 1, ZSTR_DUPLICATE);
  92. w->errorno = errorno;
  93. return w;
  94. }
  95. /* }}} */
  96. /* {{{ MYSQLI_WARNING *php_get_warnings(MYSQL *mysql) */
  97. MYSQLI_WARNING * php_get_warnings(MYSQLND_CONN_DATA * mysql)
  98. {
  99. MYSQLI_WARNING *w, *first = NULL, *prev = NULL;
  100. MYSQL_RES *result;
  101. zval row;
  102. if (mysql->m->query(mysql, "SHOW WARNINGS", 13)) {
  103. return NULL;
  104. }
  105. result = mysql->m->use_result(mysql, 0);
  106. for (;;) {
  107. zval *entry;
  108. int errno;
  109. mysqlnd_fetch_into(result, MYSQLND_FETCH_NUM, &row, MYSQLND_MYSQLI);
  110. if (Z_TYPE(row) != IS_ARRAY) {
  111. zval_ptr_dtor(&row);
  112. break;
  113. }
  114. zend_hash_internal_pointer_reset(Z_ARRVAL(row));
  115. /* 0. we don't care about the first */
  116. zend_hash_move_forward(Z_ARRVAL(row));
  117. /* 1. Here comes the error no */
  118. entry = zend_hash_get_current_data(Z_ARRVAL(row));
  119. errno = zval_get_long(entry);
  120. zend_hash_move_forward(Z_ARRVAL(row));
  121. /* 2. Here comes the reason */
  122. entry = zend_hash_get_current_data(Z_ARRVAL(row));
  123. w = php_new_warning(entry, errno);
  124. /*
  125. Don't destroy entry, because the row destroy will decrease
  126. the refcounter. Decreased twice then mysqlnd_free_result()
  127. will crash, because it will try to access already freed memory.
  128. */
  129. if (!first) {
  130. first = w;
  131. }
  132. if (prev) {
  133. prev->next = (void *)w;
  134. }
  135. prev = w;
  136. zval_ptr_dtor(&row);
  137. }
  138. mysql_free_result(result);
  139. return first;
  140. }
  141. /* }}} */
  142. #endif
  143. /* {{{ bool mysqli_warning::next() */
  144. PHP_METHOD(mysqli_warning, next)
  145. {
  146. MYSQLI_WARNING *w;
  147. zval *mysqli_warning;
  148. mysqli_object *obj = Z_MYSQLI_P(getThis());
  149. if (obj->ptr) {
  150. if (zend_parse_method_parameters(ZEND_NUM_ARGS(), getThis(), "O",
  151. &mysqli_warning, mysqli_warning_class_entry) == FAILURE) {
  152. return;
  153. }
  154. MYSQLI_FETCH_RESOURCE(w, MYSQLI_WARNING *, mysqli_warning, "mysqli_warning", MYSQLI_STATUS_VALID);
  155. if (w && w->next) {
  156. w = w->next;
  157. ((MYSQLI_RESOURCE *)(obj->ptr))->ptr = w;
  158. RETURN_TRUE;
  159. }
  160. }
  161. RETURN_FALSE;
  162. }
  163. /* }}} */
  164. /* {{{ property mysqli_warning_message */
  165. static
  166. zval *mysqli_warning_message(mysqli_object *obj, zval *retval)
  167. {
  168. MYSQLI_WARNING *w;
  169. if (!obj->ptr || !((MYSQLI_RESOURCE *)(obj->ptr))->ptr) {
  170. return NULL;
  171. }
  172. w = (MYSQLI_WARNING *)((MYSQLI_RESOURCE *)(obj->ptr))->ptr;
  173. ZVAL_COPY(retval, &w->reason);
  174. return retval;
  175. }
  176. /* }}} */
  177. /* {{{ property mysqli_warning_sqlstate */
  178. static
  179. zval *mysqli_warning_sqlstate(mysqli_object *obj, zval *retval)
  180. {
  181. MYSQLI_WARNING *w;
  182. if (!obj->ptr || !((MYSQLI_RESOURCE *)(obj->ptr))->ptr) {
  183. return NULL;
  184. }
  185. w = (MYSQLI_WARNING *)((MYSQLI_RESOURCE *)(obj->ptr))->ptr;
  186. ZVAL_COPY(retval, &w->sqlstate);
  187. return retval;
  188. }
  189. /* }}} */
  190. /* {{{ property mysqli_warning_error */
  191. static
  192. zval *mysqli_warning_errno(mysqli_object *obj, zval *retval)
  193. {
  194. MYSQLI_WARNING *w;
  195. if (!obj->ptr || !((MYSQLI_RESOURCE *)(obj->ptr))->ptr) {
  196. return NULL;
  197. }
  198. w = (MYSQLI_WARNING *)((MYSQLI_RESOURCE *)(obj->ptr))->ptr;
  199. ZVAL_LONG(retval, w->errorno);
  200. return retval;
  201. }
  202. /* }}} */
  203. /* {{{ mysqli_warning_construct(object obj) */
  204. PHP_METHOD(mysqli_warning, __construct)
  205. {
  206. zval *z;
  207. mysqli_object *obj;
  208. #ifndef MYSQLI_USE_MYSQLND
  209. MYSQL *hdl;
  210. #endif
  211. MYSQLI_WARNING *w;
  212. MYSQLI_RESOURCE *mysqli_resource;
  213. if (zend_parse_parameters(ZEND_NUM_ARGS(), "o", &z) == FAILURE) {
  214. return;
  215. }
  216. obj = Z_MYSQLI_P(z);
  217. if (obj->zo.ce == mysqli_link_class_entry) {
  218. MY_MYSQL *mysql;
  219. MYSQLI_FETCH_RESOURCE_CONN(mysql, z, MYSQLI_STATUS_VALID);
  220. if (mysql_warning_count(mysql->mysql)) {
  221. #ifndef MYSQLI_USE_MYSQLND
  222. w = php_get_warnings(mysql->mysql);
  223. #else
  224. w = php_get_warnings(mysql->mysql->data);
  225. #endif
  226. } else {
  227. php_error_docref(NULL, E_WARNING, "No warnings found");
  228. RETURN_FALSE;
  229. }
  230. } else if (obj->zo.ce == mysqli_stmt_class_entry) {
  231. MY_STMT *stmt;
  232. MYSQLI_FETCH_RESOURCE_STMT(stmt, z, MYSQLI_STATUS_VALID);
  233. #ifndef MYSQLI_USE_MYSQLND
  234. hdl = mysqli_stmt_get_connection(stmt->stmt);
  235. if (mysql_warning_count(hdl)) {
  236. w = php_get_warnings(hdl);
  237. #else
  238. if (mysqlnd_stmt_warning_count(stmt->stmt)) {
  239. w = php_get_warnings(mysqli_stmt_get_connection(stmt->stmt));
  240. #endif
  241. } else {
  242. php_error_docref(NULL, E_WARNING, "No warnings found");
  243. RETURN_FALSE;
  244. }
  245. } else {
  246. php_error_docref(NULL, E_WARNING, "invalid class argument");
  247. RETURN_FALSE;
  248. }
  249. mysqli_resource = (MYSQLI_RESOURCE *)ecalloc (1, sizeof(MYSQLI_RESOURCE));
  250. mysqli_resource->ptr = mysqli_resource->info = (void *)w;
  251. mysqli_resource->status = MYSQLI_STATUS_VALID;
  252. if (!getThis() || !instanceof_function(Z_OBJCE_P(getThis()), mysqli_warning_class_entry)) {
  253. MYSQLI_RETURN_RESOURCE(mysqli_resource, mysqli_warning_class_entry);
  254. } else {
  255. (Z_MYSQLI_P(getThis()))->ptr = mysqli_resource;
  256. }
  257. }
  258. /* }}} */
  259. /* {{{ mysqli_warning_methods */
  260. const zend_function_entry mysqli_warning_methods[] = {
  261. PHP_ME(mysqli_warning, __construct, NULL, ZEND_ACC_PROTECTED)
  262. PHP_ME(mysqli_warning, next, NULL, ZEND_ACC_PUBLIC)
  263. PHP_FE_END
  264. };
  265. /* }}} */
  266. /* {{{ mysqli_warning_property_entries */
  267. const mysqli_property_entry mysqli_warning_property_entries[] = {
  268. {"message", sizeof("message") - 1, mysqli_warning_message, NULL},
  269. {"sqlstate", sizeof("sqlstate") - 1, mysqli_warning_sqlstate, NULL},
  270. {"errno", sizeof("errno") - 1, mysqli_warning_errno, NULL},
  271. {NULL, 0, NULL, NULL}
  272. };
  273. /* }}} */
  274. /*
  275. * Local variables:
  276. * tab-width: 4
  277. * c-basic-offset: 4
  278. * indent-tabs-mode: t
  279. * End:
  280. * vim600: noet sw=4 ts=4 fdm=marker
  281. * vim<600: noet sw=4 ts=4
  282. */