129 lines
3.7 KiB
C
129 lines
3.7 KiB
C
/* Copyright Fedor Indutny. All rights reserved.
|
|
*
|
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
* of this software and associated documentation files (the "Software"), to
|
|
* deal in the Software without restriction, including without limitation the
|
|
* rights to use, copy, modify, merge, publish, distribute, sublicense, and/or
|
|
* sell copies of the Software, and to permit persons to whom the Software is
|
|
* furnished to do so, subject to the following conditions:
|
|
*
|
|
* The above copyright notice and this permission notice shall be included in
|
|
* all copies or substantial portions of the Software.
|
|
*
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
|
|
* FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
|
|
* IN THE SOFTWARE.
|
|
*/
|
|
#include "http_parser.h"
|
|
#include <assert.h>
|
|
#include <stdint.h>
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
#include <sys/time.h>
|
|
|
|
/* 8 gb */
|
|
static const int64_t kBytes = 8LL << 30;
|
|
|
|
static const char data[] =
|
|
"POST /joyent/http-parser HTTP/1.1\r\n"
|
|
"Host: github.com\r\n"
|
|
"DNT: 1\r\n"
|
|
"Accept-Encoding: gzip, deflate, sdch\r\n"
|
|
"Accept-Language: ru-RU,ru;q=0.8,en-US;q=0.6,en;q=0.4\r\n"
|
|
"User-Agent: Mozilla/5.0 (Macintosh; Intel Mac OS X 10_10_1) "
|
|
"AppleWebKit/537.36 (KHTML, like Gecko) "
|
|
"Chrome/39.0.2171.65 Safari/537.36\r\n"
|
|
"Accept: text/html,application/xhtml+xml,application/xml;q=0.9,"
|
|
"image/webp,*/*;q=0.8\r\n"
|
|
"Referer: https://github.com/joyent/http-parser\r\n"
|
|
"Connection: keep-alive\r\n"
|
|
"Transfer-Encoding: chunked\r\n"
|
|
"Cache-Control: max-age=0\r\n\r\nb\r\nhello world\r\n0\r\n";
|
|
static const size_t data_len = sizeof(data) - 1;
|
|
|
|
static int on_info(http_parser* p) {
|
|
return 0;
|
|
}
|
|
|
|
|
|
static int on_data(http_parser* p, const char *at, size_t length) {
|
|
return 0;
|
|
}
|
|
|
|
static http_parser_settings settings = {
|
|
.on_message_begin = on_info,
|
|
.on_headers_complete = on_info,
|
|
.on_message_complete = on_info,
|
|
.on_header_field = on_data,
|
|
.on_header_value = on_data,
|
|
.on_url = on_data,
|
|
.on_status = on_data,
|
|
.on_body = on_data
|
|
};
|
|
|
|
int bench(int iter_count, int silent) {
|
|
struct http_parser parser;
|
|
int i;
|
|
int err;
|
|
struct timeval start;
|
|
struct timeval end;
|
|
|
|
if (!silent) {
|
|
err = gettimeofday(&start, NULL);
|
|
assert(err == 0);
|
|
}
|
|
|
|
fprintf(stderr, "req_len=%d\n", (int) data_len);
|
|
for (i = 0; i < iter_count; i++) {
|
|
size_t parsed;
|
|
http_parser_init(&parser, HTTP_REQUEST);
|
|
|
|
parsed = http_parser_execute(&parser, &settings, data, data_len);
|
|
assert(parsed == data_len);
|
|
}
|
|
|
|
if (!silent) {
|
|
double elapsed;
|
|
double bw;
|
|
double total;
|
|
|
|
err = gettimeofday(&end, NULL);
|
|
assert(err == 0);
|
|
|
|
fprintf(stdout, "Benchmark result:\n");
|
|
|
|
elapsed = (double) (end.tv_sec - start.tv_sec) +
|
|
(end.tv_usec - start.tv_usec) * 1e-6f;
|
|
|
|
total = (double) iter_count * data_len;
|
|
bw = (double) total / elapsed;
|
|
|
|
fprintf(stdout, "%.2f mb | %.2f mb/s | %.2f req/sec | %.2f s\n",
|
|
(double) total / (1024 * 1024),
|
|
bw / (1024 * 1024),
|
|
(double) iter_count / elapsed,
|
|
elapsed);
|
|
|
|
fflush(stdout);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int main(int argc, char** argv) {
|
|
int64_t iterations;
|
|
|
|
iterations = kBytes / (int64_t) data_len;
|
|
if (argc == 2 && strcmp(argv[1], "infinite") == 0) {
|
|
for (;;)
|
|
bench(iterations, 1);
|
|
return 0;
|
|
} else {
|
|
return bench(iterations, 0);
|
|
}
|
|
}
|