mysqlnd_debug.c 24 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792
  1. /*
  2. +----------------------------------------------------------------------+
  3. | PHP Version 7 |
  4. +----------------------------------------------------------------------+
  5. | Copyright (c) 2006-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. | Authors: Andrey Hristov <andrey@php.net> |
  16. | Ulf Wendel <uw@php.net> |
  17. +----------------------------------------------------------------------+
  18. */
  19. #include "php.h"
  20. #include "mysqlnd.h"
  21. #include "mysqlnd_priv.h"
  22. #include "mysqlnd_debug.h"
  23. static const char * const mysqlnd_debug_default_trace_file = "/tmp/mysqlnd.trace";
  24. static const char * const mysqlnd_debug_empty_string = "";
  25. /* {{{ mysqlnd_debug::open */
  26. static enum_func_status
  27. MYSQLND_METHOD(mysqlnd_debug, open)(MYSQLND_DEBUG * self, zend_bool reopen)
  28. {
  29. if (!self->file_name) {
  30. return FAIL;
  31. }
  32. self->stream = php_stream_open_wrapper(self->file_name,
  33. reopen == TRUE || self->flags & MYSQLND_DEBUG_APPEND? "ab":"wb",
  34. REPORT_ERRORS, NULL);
  35. return self->stream? PASS:FAIL;
  36. }
  37. /* }}} */
  38. /* {{{ mysqlnd_debug::log */
  39. static enum_func_status
  40. MYSQLND_METHOD(mysqlnd_debug, log)(MYSQLND_DEBUG * self,
  41. unsigned int line, const char * const file,
  42. unsigned int level, const char * type, const char * message)
  43. {
  44. char pipe_buffer[512];
  45. enum_func_status ret;
  46. int i;
  47. char * message_line;
  48. unsigned int message_line_len;
  49. unsigned int flags = self->flags;
  50. char pid_buffer[10], time_buffer[30], file_buffer[200],
  51. line_buffer[6], level_buffer[7];
  52. if (!self->stream && FAIL == self->m->open(self, FALSE)) {
  53. return FAIL;
  54. }
  55. if (level == -1) {
  56. level = zend_stack_count(&self->call_stack);
  57. }
  58. i = MIN(level, sizeof(pipe_buffer) / 2 - 1);
  59. pipe_buffer[i*2] = '\0';
  60. for (;i > 0;i--) {
  61. pipe_buffer[i*2 - 1] = ' ';
  62. pipe_buffer[i*2 - 2] = '|';
  63. }
  64. if (flags & MYSQLND_DEBUG_DUMP_PID) {
  65. snprintf(pid_buffer, sizeof(pid_buffer) - 1, "%5u: ", self->pid);
  66. pid_buffer[sizeof(pid_buffer) - 1 ] = '\0';
  67. }
  68. if (flags & MYSQLND_DEBUG_DUMP_TIME) {
  69. /* The following from FF's DBUG library, which is in the public domain */
  70. #if defined(PHP_WIN32)
  71. /* FIXME This doesn't give microseconds as in Unix case, and the resolution is
  72. in system ticks, 10 ms intervals. See my_getsystime.c for high res */
  73. SYSTEMTIME loc_t;
  74. GetLocalTime(&loc_t);
  75. snprintf(time_buffer, sizeof(time_buffer) - 1,
  76. /* "%04d-%02d-%02d " */
  77. "%02d:%02d:%02d.%06d ",
  78. /*tm_p->tm_year + 1900, tm_p->tm_mon + 1, tm_p->tm_mday,*/
  79. loc_t.wHour, loc_t.wMinute, loc_t.wSecond, loc_t.wMilliseconds);
  80. time_buffer[sizeof(time_buffer) - 1 ] = '\0';
  81. #else
  82. struct timeval tv;
  83. struct tm *tm_p;
  84. if (gettimeofday(&tv, NULL) != -1) {
  85. if ((tm_p= localtime((const time_t *)&tv.tv_sec))) {
  86. snprintf(time_buffer, sizeof(time_buffer) - 1,
  87. /* "%04d-%02d-%02d " */
  88. "%02d:%02d:%02d.%06d ",
  89. /*tm_p->tm_year + 1900, tm_p->tm_mon + 1, tm_p->tm_mday,*/
  90. tm_p->tm_hour, tm_p->tm_min, tm_p->tm_sec,
  91. (int) (tv.tv_usec));
  92. time_buffer[sizeof(time_buffer) - 1 ] = '\0';
  93. }
  94. }
  95. #endif
  96. }
  97. if (flags & MYSQLND_DEBUG_DUMP_FILE) {
  98. snprintf(file_buffer, sizeof(file_buffer) - 1, "%14s: ", file);
  99. file_buffer[sizeof(file_buffer) - 1 ] = '\0';
  100. }
  101. if (flags & MYSQLND_DEBUG_DUMP_LINE) {
  102. snprintf(line_buffer, sizeof(line_buffer) - 1, "%5u: ", line);
  103. line_buffer[sizeof(line_buffer) - 1 ] = '\0';
  104. }
  105. if (flags & MYSQLND_DEBUG_DUMP_LEVEL) {
  106. snprintf(level_buffer, sizeof(level_buffer) - 1, "%4u: ", level);
  107. level_buffer[sizeof(level_buffer) - 1 ] = '\0';
  108. }
  109. message_line_len = mnd_sprintf(&message_line, 0, "%s%s%s%s%s%s%s%s\n",
  110. flags & MYSQLND_DEBUG_DUMP_PID? pid_buffer:"",
  111. flags & MYSQLND_DEBUG_DUMP_TIME? time_buffer:"",
  112. flags & MYSQLND_DEBUG_DUMP_FILE? file_buffer:"",
  113. flags & MYSQLND_DEBUG_DUMP_LINE? line_buffer:"",
  114. flags & MYSQLND_DEBUG_DUMP_LEVEL? level_buffer:"",
  115. pipe_buffer, type? type:"", message);
  116. ret = php_stream_write(self->stream, message_line, message_line_len)? PASS:FAIL;
  117. mnd_sprintf_free(message_line);
  118. if (flags & MYSQLND_DEBUG_FLUSH) {
  119. self->m->close(self);
  120. self->m->open(self, TRUE);
  121. }
  122. return ret;
  123. }
  124. /* }}} */
  125. /* {{{ mysqlnd_debug::log_va */
  126. static enum_func_status
  127. MYSQLND_METHOD(mysqlnd_debug, log_va)(MYSQLND_DEBUG *self,
  128. unsigned int line, const char * const file,
  129. unsigned int level, const char * type,
  130. const char *format, ...)
  131. {
  132. char pipe_buffer[512];
  133. int i;
  134. enum_func_status ret;
  135. char * message_line, *buffer;
  136. unsigned int message_line_len;
  137. va_list args;
  138. unsigned int flags = self->flags;
  139. char pid_buffer[10], time_buffer[30], file_buffer[200],
  140. line_buffer[6], level_buffer[7];
  141. if (!self->stream && FAIL == self->m->open(self, FALSE)) {
  142. return FAIL;
  143. }
  144. if (level == -1) {
  145. level = zend_stack_count(&self->call_stack);
  146. }
  147. i = MIN(level, sizeof(pipe_buffer) / 2 - 1);
  148. pipe_buffer[i*2] = '\0';
  149. for (;i > 0;i--) {
  150. pipe_buffer[i*2 - 1] = ' ';
  151. pipe_buffer[i*2 - 2] = '|';
  152. }
  153. if (flags & MYSQLND_DEBUG_DUMP_PID) {
  154. snprintf(pid_buffer, sizeof(pid_buffer) - 1, "%5u: ", self->pid);
  155. pid_buffer[sizeof(pid_buffer) - 1 ] = '\0';
  156. }
  157. if (flags & MYSQLND_DEBUG_DUMP_TIME) {
  158. /* The following from FF's DBUG library, which is in the public domain */
  159. #if defined(PHP_WIN32)
  160. /* FIXME This doesn't give microseconds as in Unix case, and the resolution is
  161. in system ticks, 10 ms intervals. See my_getsystime.c for high res */
  162. SYSTEMTIME loc_t;
  163. GetLocalTime(&loc_t);
  164. snprintf(time_buffer, sizeof(time_buffer) - 1,
  165. /* "%04d-%02d-%02d " */
  166. "%02d:%02d:%02d.%06d ",
  167. /*tm_p->tm_year + 1900, tm_p->tm_mon + 1, tm_p->tm_mday,*/
  168. loc_t.wHour, loc_t.wMinute, loc_t.wSecond, loc_t.wMilliseconds);
  169. time_buffer[sizeof(time_buffer) - 1 ] = '\0';
  170. #else
  171. struct timeval tv;
  172. struct tm *tm_p;
  173. if (gettimeofday(&tv, NULL) != -1) {
  174. if ((tm_p= localtime((const time_t *)&tv.tv_sec))) {
  175. snprintf(time_buffer, sizeof(time_buffer) - 1,
  176. /* "%04d-%02d-%02d " */
  177. "%02d:%02d:%02d.%06d ",
  178. /*tm_p->tm_year + 1900, tm_p->tm_mon + 1, tm_p->tm_mday,*/
  179. tm_p->tm_hour, tm_p->tm_min, tm_p->tm_sec,
  180. (int) (tv.tv_usec));
  181. time_buffer[sizeof(time_buffer) - 1 ] = '\0';
  182. }
  183. }
  184. #endif
  185. }
  186. if (flags & MYSQLND_DEBUG_DUMP_FILE) {
  187. snprintf(file_buffer, sizeof(file_buffer) - 1, "%14s: ", file);
  188. file_buffer[sizeof(file_buffer) - 1 ] = '\0';
  189. }
  190. if (flags & MYSQLND_DEBUG_DUMP_LINE) {
  191. snprintf(line_buffer, sizeof(line_buffer) - 1, "%5u: ", line);
  192. line_buffer[sizeof(line_buffer) - 1 ] = '\0';
  193. }
  194. if (flags & MYSQLND_DEBUG_DUMP_LEVEL) {
  195. snprintf(level_buffer, sizeof(level_buffer) - 1, "%4u: ", level);
  196. level_buffer[sizeof(level_buffer) - 1 ] = '\0';
  197. }
  198. va_start(args, format);
  199. mnd_vsprintf(&buffer, 0, format, args);
  200. va_end(args);
  201. message_line_len = mnd_sprintf(&message_line, 0, "%s%s%s%s%s%s%s%s\n",
  202. flags & MYSQLND_DEBUG_DUMP_PID? pid_buffer:"",
  203. flags & MYSQLND_DEBUG_DUMP_TIME? time_buffer:"",
  204. flags & MYSQLND_DEBUG_DUMP_FILE? file_buffer:"",
  205. flags & MYSQLND_DEBUG_DUMP_LINE? line_buffer:"",
  206. flags & MYSQLND_DEBUG_DUMP_LEVEL? level_buffer:"",
  207. pipe_buffer, type? type:"", buffer);
  208. mnd_sprintf_free(buffer);
  209. ret = php_stream_write(self->stream, message_line, message_line_len)? PASS:FAIL;
  210. mnd_sprintf_free(message_line);
  211. if (flags & MYSQLND_DEBUG_FLUSH) {
  212. self->m->close(self);
  213. self->m->open(self, TRUE);
  214. }
  215. return ret;
  216. }
  217. /* }}} */
  218. /* FALSE - The DBG_ calls won't be traced, TRUE - will be traced */
  219. /* {{{ mysqlnd_debug::func_enter */
  220. static zend_bool
  221. MYSQLND_METHOD(mysqlnd_debug, func_enter)(MYSQLND_DEBUG * self,
  222. unsigned int line, const char * const file,
  223. const char * const func_name, unsigned int func_name_len)
  224. {
  225. if ((self->flags & MYSQLND_DEBUG_DUMP_TRACE) == 0 || self->file_name == NULL) {
  226. return FALSE;
  227. }
  228. if ((uint32_t) zend_stack_count(&self->call_stack) >= self->nest_level_limit) {
  229. return FALSE;
  230. }
  231. if ((self->flags & MYSQLND_DEBUG_TRACE_MEMORY_CALLS) == 0 && self->skip_functions) {
  232. const char ** p = self->skip_functions;
  233. while (*p) {
  234. if (*p == func_name) {
  235. zend_stack_push(&self->call_stack, &mysqlnd_debug_empty_string);
  236. #ifndef MYSQLND_PROFILING_DISABLED
  237. if (self->flags & MYSQLND_DEBUG_PROFILE_CALLS) {
  238. uint64_t some_time = 0;
  239. zend_stack_push(&self->call_time_stack, &some_time);
  240. }
  241. #endif
  242. return FALSE;
  243. }
  244. p++;
  245. }
  246. }
  247. zend_stack_push(&self->call_stack, &func_name);
  248. #ifndef MYSQLND_PROFILING_DISABLED
  249. if (self->flags & MYSQLND_DEBUG_PROFILE_CALLS) {
  250. uint64_t some_time = 0;
  251. zend_stack_push(&self->call_time_stack, &some_time);
  252. }
  253. #endif
  254. if (zend_hash_num_elements(&self->not_filtered_functions) &&
  255. 0 == zend_hash_str_exists(&self->not_filtered_functions, func_name, strlen(func_name)))
  256. {
  257. return FALSE;
  258. }
  259. self->m->log_va(self, line, file, zend_stack_count(&self->call_stack) - 1, NULL, ">%s", func_name);
  260. return TRUE;
  261. }
  262. /* }}} */
  263. #ifndef MYSQLND_PROFILING_DISABLED
  264. struct st_mysqlnd_dbg_function_profile {
  265. uint64_t calls;
  266. uint64_t min_own;
  267. uint64_t max_own;
  268. uint64_t avg_own;
  269. uint64_t own_underporm_calls;
  270. uint64_t min_in_calls;
  271. uint64_t max_in_calls;
  272. uint64_t avg_in_calls;
  273. uint64_t in_calls_underporm_calls;
  274. uint64_t min_total;
  275. uint64_t max_total;
  276. uint64_t avg_total;
  277. uint64_t total_underporm_calls;
  278. };
  279. #define PROFILE_UNDERPERFORM_THRESHOLD 10
  280. #endif
  281. /* {{{ mysqlnd_debug::func_leave */
  282. static enum_func_status
  283. MYSQLND_METHOD(mysqlnd_debug, func_leave)(MYSQLND_DEBUG * self, unsigned int line, const char * const file, uint64_t call_time)
  284. {
  285. char **func_name;
  286. uint64_t * parent_non_own_time_ptr = NULL, * mine_non_own_time_ptr = NULL;
  287. uint64_t mine_non_own_time = 0;
  288. zend_bool profile_calls = self->flags & MYSQLND_DEBUG_PROFILE_CALLS? TRUE:FALSE;
  289. if ((self->flags & MYSQLND_DEBUG_DUMP_TRACE) == 0 || self->file_name == NULL) {
  290. return PASS;
  291. }
  292. if ((uint32_t) zend_stack_count(&self->call_stack) >= self->nest_level_limit) {
  293. return PASS;
  294. }
  295. func_name = zend_stack_top(&self->call_stack);
  296. #ifndef MYSQLND_PROFILING_DISABLED
  297. if (profile_calls) {
  298. mine_non_own_time_ptr = zend_stack_top(&self->call_time_stack);
  299. mine_non_own_time = *mine_non_own_time_ptr;
  300. zend_stack_del_top(&self->call_time_stack); /* callee - removing ourselves */
  301. }
  302. #endif
  303. if ((*func_name)[0] == '\0') {
  304. ; /* don't log that function */
  305. } else if (!zend_hash_num_elements(&self->not_filtered_functions) ||
  306. 1 == zend_hash_str_exists(&self->not_filtered_functions, (*func_name), strlen((*func_name))))
  307. {
  308. #ifndef MYSQLND_PROFILING_DISABLED
  309. if (FALSE == profile_calls) {
  310. #endif
  311. self->m->log_va(self, line, file, zend_stack_count(&self->call_stack) - 1, NULL, "<%s", *func_name);
  312. #ifndef MYSQLND_PROFILING_DISABLED
  313. } else {
  314. struct st_mysqlnd_dbg_function_profile f_profile_stack = {0};
  315. struct st_mysqlnd_dbg_function_profile * f_profile = NULL;
  316. uint64_t own_time = call_time - mine_non_own_time;
  317. uint32_t func_name_len = strlen(*func_name);
  318. self->m->log_va(self, line, file, zend_stack_count(&self->call_stack) - 1, NULL, "<%s (total=%u own=%u in_calls=%u)",
  319. *func_name, (unsigned int) call_time, (unsigned int) own_time, (unsigned int) mine_non_own_time
  320. );
  321. if ((f_profile = zend_hash_str_find_ptr(&self->function_profiles, *func_name, func_name_len)) != NULL) {
  322. /* found */
  323. if (f_profile) {
  324. if (mine_non_own_time < f_profile->min_in_calls) {
  325. f_profile->min_in_calls = mine_non_own_time;
  326. } else if (mine_non_own_time > f_profile->max_in_calls) {
  327. f_profile->max_in_calls = mine_non_own_time;
  328. }
  329. f_profile->avg_in_calls = (f_profile->avg_in_calls * f_profile->calls + mine_non_own_time) / (f_profile->calls + 1);
  330. if (own_time < f_profile->min_own) {
  331. f_profile->min_own = own_time;
  332. } else if (own_time > f_profile->max_own) {
  333. f_profile->max_own = own_time;
  334. }
  335. f_profile->avg_own = (f_profile->avg_own * f_profile->calls + own_time) / (f_profile->calls + 1);
  336. if (call_time < f_profile->min_total) {
  337. f_profile->min_total = call_time;
  338. } else if (call_time > f_profile->max_total) {
  339. f_profile->max_total = call_time;
  340. }
  341. f_profile->avg_total = (f_profile->avg_total * f_profile->calls + call_time) / (f_profile->calls + 1);
  342. ++f_profile->calls;
  343. if (f_profile->calls > PROFILE_UNDERPERFORM_THRESHOLD) {
  344. if (f_profile->avg_in_calls < mine_non_own_time) {
  345. f_profile->in_calls_underporm_calls++;
  346. }
  347. if (f_profile->avg_own < own_time) {
  348. f_profile->own_underporm_calls++;
  349. }
  350. if (f_profile->avg_total < call_time) {
  351. f_profile->total_underporm_calls++;
  352. }
  353. }
  354. }
  355. } else {
  356. /* add */
  357. f_profile = &f_profile_stack;
  358. f_profile->min_in_calls = f_profile->max_in_calls = f_profile->avg_in_calls = mine_non_own_time;
  359. f_profile->min_total = f_profile->max_total = f_profile->avg_total = call_time;
  360. f_profile->min_own = f_profile->max_own = f_profile->avg_own = own_time;
  361. f_profile->calls = 1;
  362. zend_hash_str_add_mem(&self->function_profiles, *func_name, func_name_len, f_profile, sizeof(struct st_mysqlnd_dbg_function_profile));
  363. }
  364. if ((uint32_t) zend_stack_count(&self->call_time_stack)) {
  365. uint64_t parent_non_own_time = 0;
  366. parent_non_own_time_ptr = zend_stack_top(&self->call_time_stack);
  367. parent_non_own_time = *parent_non_own_time_ptr;
  368. parent_non_own_time += call_time;
  369. zend_stack_del_top(&self->call_time_stack); /* the caller */
  370. zend_stack_push(&self->call_time_stack, &parent_non_own_time); /* add back the caller */
  371. }
  372. }
  373. #endif
  374. }
  375. return zend_stack_del_top(&self->call_stack) == SUCCESS? PASS:FAIL;
  376. }
  377. /* }}} */
  378. /* {{{ mysqlnd_debug::close */
  379. static enum_func_status
  380. MYSQLND_METHOD(mysqlnd_debug, close)(MYSQLND_DEBUG * self)
  381. {
  382. if (self->stream) {
  383. #ifndef MYSQLND_PROFILING_DISABLED
  384. if (!(self->flags & MYSQLND_DEBUG_FLUSH) && (self->flags & MYSQLND_DEBUG_PROFILE_CALLS)) {
  385. struct st_mysqlnd_dbg_function_profile * f_profile;
  386. zend_string *string_key = NULL;
  387. self->m->log_va(self, __LINE__, __FILE__, 0, "info : ",
  388. "number of functions: %d", zend_hash_num_elements(&self->function_profiles));
  389. ZEND_HASH_FOREACH_STR_KEY_PTR(&self->function_profiles, string_key, f_profile) {
  390. self->m->log_va(self, __LINE__, __FILE__, -1, "info : ",
  391. "%-40s\tcalls=%5llu own_slow=%5llu in_calls_slow=%5llu total_slow=%5llu"
  392. " min_own=%5llu max_own=%7llu avg_own=%7llu "
  393. " min_in_calls=%5llu max_in_calls=%7llu avg_in_calls=%7llu"
  394. " min_total=%5llu max_total=%7llu avg_total=%7llu"
  395. ,ZSTR_VAL(string_key)
  396. ,(uint64_t) f_profile->calls
  397. ,(uint64_t) f_profile->own_underporm_calls
  398. ,(uint64_t) f_profile->in_calls_underporm_calls
  399. ,(uint64_t) f_profile->total_underporm_calls
  400. ,(uint64_t) f_profile->min_own
  401. ,(uint64_t) f_profile->max_own
  402. ,(uint64_t) f_profile->avg_own
  403. ,(uint64_t) f_profile->min_in_calls
  404. ,(uint64_t) f_profile->max_in_calls
  405. ,(uint64_t) f_profile->avg_in_calls
  406. ,(uint64_t) f_profile->min_total
  407. ,(uint64_t) f_profile->max_total
  408. ,(uint64_t) f_profile->avg_total
  409. );
  410. } ZEND_HASH_FOREACH_END();
  411. }
  412. #endif
  413. php_stream_close(self->stream);
  414. self->stream = NULL;
  415. }
  416. /* no DBG_RETURN please */
  417. return PASS;
  418. }
  419. /* }}} */
  420. /* {{{ mysqlnd_res_meta::free */
  421. static enum_func_status
  422. MYSQLND_METHOD(mysqlnd_debug, free)(MYSQLND_DEBUG * self)
  423. {
  424. if (self->file_name && self->file_name != mysqlnd_debug_default_trace_file) {
  425. efree(self->file_name);
  426. self->file_name = NULL;
  427. }
  428. zend_stack_destroy(&self->call_stack);
  429. zend_stack_destroy(&self->call_time_stack);
  430. zend_hash_destroy(&self->not_filtered_functions);
  431. zend_hash_destroy(&self->function_profiles);
  432. free(self);
  433. return PASS;
  434. }
  435. /* }}} */
  436. enum mysqlnd_debug_parser_state
  437. {
  438. PARSER_WAIT_MODIFIER,
  439. PARSER_WAIT_COLON,
  440. PARSER_WAIT_VALUE
  441. };
  442. /* {{{ mysqlnd_res_meta::set_mode */
  443. static void
  444. MYSQLND_METHOD(mysqlnd_debug, set_mode)(MYSQLND_DEBUG * self, const char * const mode)
  445. {
  446. unsigned int mode_len, i;
  447. enum mysqlnd_debug_parser_state state = PARSER_WAIT_MODIFIER;
  448. mode_len = mode? strlen(mode) : 0;
  449. self->flags = 0;
  450. self->nest_level_limit = 0;
  451. if (self->file_name && self->file_name != mysqlnd_debug_default_trace_file) {
  452. efree(self->file_name);
  453. self->file_name = NULL;
  454. }
  455. if (zend_hash_num_elements(&self->not_filtered_functions)) {
  456. zend_hash_destroy(&self->not_filtered_functions);
  457. zend_hash_init(&self->not_filtered_functions, 0, NULL, NULL, 0);
  458. }
  459. for (i = 0; i < mode_len; i++) {
  460. switch (mode[i]) {
  461. case 'O':
  462. case 'A':
  463. self->flags |= MYSQLND_DEBUG_FLUSH;
  464. case 'a':
  465. case 'o':
  466. if (mode[i] == 'a' || mode[i] == 'A') {
  467. self->flags |= MYSQLND_DEBUG_APPEND;
  468. }
  469. if (i + 1 < mode_len && mode[i+1] == ',') {
  470. unsigned int j = i + 2;
  471. #ifdef PHP_WIN32
  472. if (i+4 < mode_len && mode[i+3] == ':' && (mode[i+4] == '\\' || mode[i+4] == '/')) {
  473. j = i + 5;
  474. }
  475. #endif
  476. while (j < mode_len) {
  477. if (mode[j] == ':') {
  478. break;
  479. }
  480. j++;
  481. }
  482. if (j > i + 2) {
  483. self->file_name = estrndup(mode + i + 2, j - i - 2);
  484. }
  485. i = j;
  486. } else {
  487. if (!self->file_name)
  488. self->file_name = (char *) mysqlnd_debug_default_trace_file;
  489. }
  490. state = PARSER_WAIT_COLON;
  491. break;
  492. case ':':
  493. #if 0
  494. if (state != PARSER_WAIT_COLON) {
  495. php_error_docref(NULL, E_WARNING, "Consecutive semicolons at position %u", i);
  496. }
  497. #endif
  498. state = PARSER_WAIT_MODIFIER;
  499. break;
  500. case 'f': /* limit output to these functions */
  501. if (i + 1 < mode_len && mode[i+1] == ',') {
  502. unsigned int j = i + 2;
  503. i++;
  504. while (j < mode_len) {
  505. if (mode[j] == ':') {
  506. /* function names with :: */
  507. if ((j + 1 < mode_len) && mode[j+1] == ':') {
  508. j += 2;
  509. continue;
  510. }
  511. }
  512. if (mode[j] == ',' || mode[j] == ':') {
  513. if (j > i + 2) {
  514. char func_name[1024];
  515. unsigned int func_name_len = MIN(sizeof(func_name) - 1, j - i - 1);
  516. memcpy(func_name, mode + i + 1, func_name_len);
  517. func_name[func_name_len] = '\0';
  518. zend_hash_str_add_empty_element(&self->not_filtered_functions,
  519. func_name, func_name_len);
  520. i = j;
  521. }
  522. if (mode[j] == ':') {
  523. break;
  524. }
  525. }
  526. j++;
  527. }
  528. i = j;
  529. } else {
  530. #if 0
  531. php_error_docref(NULL, E_WARNING,
  532. "Expected list of functions for '%c' found none", mode[i]);
  533. #endif
  534. }
  535. state = PARSER_WAIT_COLON;
  536. break;
  537. case 'D':
  538. case 'd':
  539. case 'g':
  540. case 'p':
  541. /* unsupported */
  542. if ((i + 1) < mode_len && mode[i+1] == ',') {
  543. i+= 2;
  544. while (i < mode_len) {
  545. if (mode[i] == ':') {
  546. break;
  547. }
  548. i++;
  549. }
  550. }
  551. state = PARSER_WAIT_COLON;
  552. break;
  553. case 'F':
  554. self->flags |= MYSQLND_DEBUG_DUMP_FILE;
  555. state = PARSER_WAIT_COLON;
  556. break;
  557. case 'i':
  558. self->flags |= MYSQLND_DEBUG_DUMP_PID;
  559. state = PARSER_WAIT_COLON;
  560. break;
  561. case 'L':
  562. self->flags |= MYSQLND_DEBUG_DUMP_LINE;
  563. state = PARSER_WAIT_COLON;
  564. break;
  565. case 'n':
  566. self->flags |= MYSQLND_DEBUG_DUMP_LEVEL;
  567. state = PARSER_WAIT_COLON;
  568. break;
  569. case 't':
  570. if (mode[i+1] == ',') {
  571. unsigned int j = i + 2;
  572. while (j < mode_len) {
  573. if (mode[j] == ':') {
  574. break;
  575. }
  576. j++;
  577. }
  578. if (j > i + 2) {
  579. char *value_str = estrndup(mode + i + 2, j - i - 2);
  580. self->nest_level_limit = atoi(value_str);
  581. efree(value_str);
  582. }
  583. i = j;
  584. } else {
  585. self->nest_level_limit = 200; /* default value for FF DBUG */
  586. }
  587. self->flags |= MYSQLND_DEBUG_DUMP_TRACE;
  588. state = PARSER_WAIT_COLON;
  589. break;
  590. case 'T':
  591. self->flags |= MYSQLND_DEBUG_DUMP_TIME;
  592. state = PARSER_WAIT_COLON;
  593. break;
  594. case 'N':
  595. case 'P':
  596. case 'r':
  597. case 'S':
  598. state = PARSER_WAIT_COLON;
  599. break;
  600. case 'm': /* mysqlnd extension - trace memory functions */
  601. self->flags |= MYSQLND_DEBUG_TRACE_MEMORY_CALLS;
  602. state = PARSER_WAIT_COLON;
  603. break;
  604. case 'x': /* mysqlnd extension - profile calls */
  605. self->flags |= MYSQLND_DEBUG_PROFILE_CALLS;
  606. state = PARSER_WAIT_COLON;
  607. break;
  608. default:
  609. if (state == PARSER_WAIT_MODIFIER) {
  610. #if 0
  611. php_error_docref(NULL, E_WARNING, "Unrecognized format '%c'", mode[i]);
  612. #endif
  613. if (i+1 < mode_len && mode[i+1] == ',') {
  614. i+= 2;
  615. while (i < mode_len) {
  616. if (mode[i] == ':') {
  617. break;
  618. }
  619. i++;
  620. }
  621. }
  622. state = PARSER_WAIT_COLON;
  623. } else if (state == PARSER_WAIT_COLON) {
  624. #if 0
  625. php_error_docref(NULL, E_WARNING, "Colon expected, '%c' found", mode[i]);
  626. #endif
  627. }
  628. break;
  629. }
  630. }
  631. }
  632. /* }}} */
  633. MYSQLND_CLASS_METHODS_START(mysqlnd_debug)
  634. MYSQLND_METHOD(mysqlnd_debug, open),
  635. MYSQLND_METHOD(mysqlnd_debug, set_mode),
  636. MYSQLND_METHOD(mysqlnd_debug, log),
  637. MYSQLND_METHOD(mysqlnd_debug, log_va),
  638. MYSQLND_METHOD(mysqlnd_debug, func_enter),
  639. MYSQLND_METHOD(mysqlnd_debug, func_leave),
  640. MYSQLND_METHOD(mysqlnd_debug, close),
  641. MYSQLND_METHOD(mysqlnd_debug, free),
  642. MYSQLND_CLASS_METHODS_END;
  643. /* {{{ mysqlnd_debug_init */
  644. PHPAPI MYSQLND_DEBUG *
  645. mysqlnd_debug_init(const char * skip_functions[])
  646. {
  647. MYSQLND_DEBUG *ret = calloc(1, sizeof(MYSQLND_DEBUG));
  648. ret->nest_level_limit = 0;
  649. ret->pid = getpid();
  650. zend_stack_init(&ret->call_stack, sizeof(char *));
  651. zend_stack_init(&ret->call_time_stack, sizeof(uint64_t));
  652. zend_hash_init(&ret->not_filtered_functions, 0, NULL, NULL, 0);
  653. zend_hash_init(&ret->function_profiles, 0, NULL, NULL, 0);
  654. ret->m = & mysqlnd_mysqlnd_debug_methods;
  655. ret->skip_functions = skip_functions;
  656. return ret;
  657. }
  658. /* }}} */
  659. /* {{{ mysqlnd_debug */
  660. PHPAPI void mysqlnd_debug(const char * mode)
  661. {
  662. #if PHP_DEBUG
  663. MYSQLND_DEBUG * dbg = MYSQLND_G(dbg);
  664. if (!dbg) {
  665. struct st_mysqlnd_plugin_trace_log * trace_log_plugin = mysqlnd_plugin_find("debug_trace");
  666. if (trace_log_plugin) {
  667. dbg = trace_log_plugin->methods.trace_instance_init(mysqlnd_debug_std_no_trace_funcs);
  668. if (!dbg) {
  669. return;
  670. }
  671. MYSQLND_G(dbg) = dbg;
  672. }
  673. }
  674. if (dbg) {
  675. dbg->m->close(dbg);
  676. dbg->m->set_mode(dbg, mode);
  677. while (zend_stack_count(&dbg->call_stack)) {
  678. zend_stack_del_top(&dbg->call_stack);
  679. }
  680. while (zend_stack_count(&dbg->call_time_stack)) {
  681. zend_stack_del_top(&dbg->call_time_stack);
  682. }
  683. }
  684. #endif
  685. }
  686. /* }}} */
  687. static struct st_mysqlnd_plugin_trace_log mysqlnd_plugin_trace_log_plugin =
  688. {
  689. {
  690. MYSQLND_PLUGIN_API_VERSION,
  691. "debug_trace",
  692. MYSQLND_VERSION_ID,
  693. PHP_MYSQLND_VERSION,
  694. "PHP License 3.01",
  695. "Andrey Hristov <andrey@php.net>, Ulf Wendel <uw@php.net>, Georg Richter <georg@php.net>",
  696. {
  697. NULL, /* no statistics , will be filled later if there are some */
  698. NULL, /* no statistics */
  699. },
  700. {
  701. NULL /* plugin shutdown */
  702. }
  703. },
  704. {/* methods */
  705. mysqlnd_debug_init,
  706. }
  707. };
  708. /* {{{ mysqlnd_debug_trace_plugin_register */
  709. void
  710. mysqlnd_debug_trace_plugin_register(void)
  711. {
  712. mysqlnd_plugin_register_ex((struct st_mysqlnd_plugin_header *) &mysqlnd_plugin_trace_log_plugin);
  713. }
  714. /* }}} */
  715. /*
  716. * Local variables:
  717. * tab-width: 4
  718. * c-basic-offset: 4
  719. * End:
  720. * vim600: noet sw=4 ts=4 fdm=marker
  721. * vim<600: noet sw=4 ts=4
  722. */