fuzz.c 4.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137
  1. #define MG_ENABLE_SOCKET 0
  2. #define MG_ENABLE_LOG 0
  3. #define MG_ENABLE_LINES 1
  4. #define MG_ENABLE_TCPIP 1
  5. #define MG_IO_SIZE (32 * 1024 * 1024) // Big IO size for fast resizes
  6. #include "mongoose.c"
  7. #include "driver_mock.c"
  8. #ifdef __cplusplus
  9. extern "C" int LLVMFuzzerTestOneInput(const uint8_t *, size_t);
  10. #else
  11. int LLVMFuzzerTestOneInput(const uint8_t *, size_t);
  12. #endif
  13. static void fn(struct mg_connection *c, int ev, void *ev_data) {
  14. struct mg_http_serve_opts opts = {.root_dir = "."};
  15. if (ev == MG_EV_HTTP_MSG) {
  16. mg_http_serve_dir(c, (struct mg_http_message *) ev_data, &opts);
  17. }
  18. }
  19. int LLVMFuzzerTestOneInput(const uint8_t *data, size_t size) {
  20. mg_log_set(MG_LL_INFO);
  21. struct mg_dns_message dm;
  22. mg_dns_parse(data, size, &dm);
  23. mg_dns_parse(NULL, 0, &dm);
  24. struct mg_http_message hm;
  25. if (mg_http_parse((const char *) data, size, &hm) > 0) {
  26. mg_crc32(0, hm.method.ptr, hm.method.len);
  27. mg_crc32(0, hm.uri.ptr, hm.uri.len);
  28. mg_crc32(0, hm.uri.ptr, hm.uri.len);
  29. for (size_t i = 0; i < sizeof(hm.headers) / sizeof(hm.headers[0]); i++) {
  30. struct mg_str *k = &hm.headers[i].name, *v = &hm.headers[i].value;
  31. mg_crc32(0, k->ptr, k->len);
  32. mg_crc32(0, v->ptr, v->len);
  33. }
  34. }
  35. mg_http_parse(NULL, 0, &hm);
  36. struct mg_str body = mg_str_n((const char *) data, size);
  37. char tmp[256];
  38. mg_http_get_var(&body, "key", tmp, sizeof(tmp));
  39. mg_http_get_var(&body, "key", NULL, 0);
  40. mg_url_decode((char *) data, size, tmp, sizeof(tmp), 1);
  41. mg_url_decode((char *) data, size, tmp, 1, 1);
  42. mg_url_decode(NULL, 0, tmp, 1, 1);
  43. struct mg_mqtt_message mm;
  44. if (mg_mqtt_parse(data, size, 0, &mm) == MQTT_OK) {
  45. mg_crc32(0, mm.topic.ptr, mm.topic.len);
  46. mg_crc32(0, mm.data.ptr, mm.data.len);
  47. mg_crc32(0, mm.dgram.ptr, mm.dgram.len);
  48. }
  49. mg_mqtt_parse(NULL, 0, 0, &mm);
  50. if (mg_mqtt_parse(data, size, 5, &mm) == MQTT_OK) {
  51. mg_crc32(0, mm.topic.ptr, mm.topic.len);
  52. mg_crc32(0, mm.data.ptr, mm.data.len);
  53. mg_crc32(0, mm.dgram.ptr, mm.dgram.len);
  54. }
  55. mg_mqtt_parse(NULL, 0, 5, &mm);
  56. mg_sntp_parse(data, size);
  57. mg_sntp_parse(NULL, 0);
  58. char buf[size * 4 / 3 + 5]; // At least 4 chars and nul termination
  59. mg_base64_decode((char *) data, size, buf, sizeof(buf));
  60. mg_base64_decode(NULL, 0, buf, sizeof(buf));
  61. mg_base64_encode(data, size, buf, sizeof(buf));
  62. mg_base64_encode(NULL, 0, buf, sizeof(buf));
  63. mg_globmatch((char *) data, size, (char *) data, size);
  64. struct mg_str k, v, s = mg_str_n((char *) data, size);
  65. while (mg_commalist(&s, &k, &v)) k.len = v.len = 0;
  66. int n;
  67. mg_json_get(mg_str_n((char *) data, size), "$", &n);
  68. mg_json_get(mg_str_n((char *) data, size), "$.a.b", &n);
  69. mg_json_get(mg_str_n((char *) data, size), "$[0]", &n);
  70. if (size > 0) {
  71. struct mg_tcpip_if mif = {.ip = 0x01020304,
  72. .mask = 255,
  73. .gw = 0x01010101,
  74. .driver = &mg_tcpip_driver_mock};
  75. struct mg_mgr mgr;
  76. mg_mgr_init(&mgr);
  77. mg_tcpip_init(&mgr, &mif);
  78. // Make a copy of the random data, in order to modify it
  79. void *pkt = malloc(size);
  80. struct eth *eth = (struct eth *) pkt;
  81. memcpy(pkt, data, size);
  82. if (size > sizeof(*eth)) {
  83. static size_t i;
  84. uint16_t eth_types[] = {0x800, 0x800, 0x806, 0x86dd};
  85. memcpy(eth->dst, mif.mac, 6); // Set valid destination MAC
  86. eth->type = mg_htons(eth_types[i++]);
  87. if (i >= sizeof(eth_types) / sizeof(eth_types[0])) i = 0;
  88. }
  89. mg_tcpip_rx(&mif, pkt, size);
  90. // Test HTTP serving
  91. const char *url = "http://localhost:12345";
  92. struct mg_connection *c = mg_http_connect(&mgr, url, fn, NULL);
  93. mg_iobuf_add(&c->recv, 0, data, size);
  94. c->pfn(c, MG_EV_READ, NULL); // manually invoke protocol event handler
  95. mg_mgr_free(&mgr);
  96. free(pkt);
  97. mg_tcpip_free(&mif);
  98. }
  99. return 0;
  100. }
  101. #if defined(MAIN)
  102. int main(int argc, char *argv[]) {
  103. int res = EXIT_FAILURE;
  104. if (argc > 1) {
  105. size_t len = 0;
  106. char *buf = mg_file_read(&mg_fs_posix, argv[1], &len);
  107. if (buf != NULL) {
  108. LLVMFuzzerTestOneInput((uint8_t *) buf, len);
  109. res = EXIT_SUCCESS;
  110. }
  111. free(buf);
  112. }
  113. return res;
  114. }
  115. #endif