summaryrefslogtreecommitdiffstatshomepage
path: root/cgit.png
blob: 0bdf5a7e72fc7d4b746527cd89e540e2b6f6c292 (plain)
ofshex dumpascii
0000 89 50 4e 47 0d 0a 1a 0a 00 00 00 0d 49 48 44 52 00 00 00 60 00 00 00 40 08 06 00 00 00 e5 34 72 .PNG........IHDR...`...@......4r
0020 0e 00 00 00 01 73 52 47 42 00 ae ce 1c e9 00 00 00 06 62 4b 47 44 00 ff 00 ff 00 ff a0 bd a7 93 .....sRGB.........bKGD..........
0040 00 00 00 09 70 48 59 73 00 00 0b 13 00 00 0b 13 01 00 9a 9c 18 00 00 05 63 49 44 41 54 78 da ed ....pHYs................cIDATx..
0060 9b 5d 88 94 55 18 c7 7f ff d9 f5 6b d7 72 dd ca 34 fa b2 34 ac 68 25 d2 28 ea 42 2b 2b ad 04 cd .]..U......k.r..4..4.h%.(.B++...
0080 99 dd 25 95 85 a0 2e a4 9b e8 22 88 88 bc f1 a6 28 a8 ab 28 4a fc 6a 76 0a a2 30 12 d1 2e 12 95 ..%.......".....(..(J.jv..0.....
00a0 52 34 cd ca 2c 8a 0a 72 43 97 5d 75 dd 4f e7 e9 e2 1d 6b 7c e7 ec ee cc be f3 91 ce f9 c3 5c cc R4..,..rC.]u.O....k|..........\.
00c0 79 e7 9d f3 9e f3 3f cf f7 f3 ca cc f0 a8 1c 62 7e 0b 3c 01 55 8d da 8b ed 81 53 52 58 67 76 c7 y.....?........b~.<.U.....SRXgv.
00e0 cd 1a bc 04 78 78 02 3c 01 1e 9e 80 8b 0d f2 71 80 f7 82 aa da 8b f2 2a c8 db 00 4f 80 87 27 c0 ....xx.<.......q.......*...O..'.
0100 13 e0 e1 09 f0 71 40 5e d8 22 35 d5 c0 22 60 1e 70 0b 30 5d d0 00 4c 00 6a 32 3f 1b 32 38 0d 9c .....q@^."5.."`.p.0]..L.j2?.28..
0120 01 8e 0b 8e 18 7c 77 0e b6 b6 9a fd 58 4e 37 d2 f1 fb a2 23 6e a6 92 c7 01 29 a9 19 78 a9 16 ee .....|w.....XN7....#n....)..x...
0140 c8 e7 7f 05 53 09 3e d7 01 f3 15 4c f6 5a 4a 3a 64 b0 36 61 f6 b1 3f ff 79 10 f0 8e 54 37 15 36 ....S.>....L.ZJ:d.6a..?.y...T7.6
0160 02 cb 8a 34 67 93 e0 a3 94 b4 b5 07 56 b4 99 f5 79 1b 30 02 a6 c2 96 22 6e 7e 36 1e ab 83 4f 52 ...4g.......V...y.0...."n~6...OR
0180 d2 78 2f 01 c3 ab 9d 55 c0 52 d7 35 83 13 82 bd 06 07 0d 4e 02 dd 82 c1 cc e5 1a 83 7a 41 a3 e0 .x/....U.R.5.......N........zA..
01a0 66 e0 7e 60 56 8e 01 82 47 0c d6 00 6f 94 6a 81 06 ab 42 73 6e 08 fd e4 ac c1 b3 ff 4b 23 dc 2e f.~`V...G...o.j...Bsn.......K#..
01c0 1d 55 60 68 b3 d1 95 86 e7 63 b0 29 6e 36 50 80 0d b9 d7 e0 3d c1 ad a1 4b 9d 03 30 f3 29 b3 53 .U`h.....c.)n6P.....=...K..0.).S
01e0 a5 30 c2 c5 be bf 6c 2a 28 29 cd 77 6c fe d0 39 78 a0 d9 ec fd 42 36 3f e3 29 ec 19 80 07 81 ce .0....l*().wl..9x....B6?.)......
0200 d0 a5 c6 71 f0 a8 b7 01 b9 ea 61 91 43 9c 53 2d 66 07 c6 3a d9 4a b3 bf 0c de 72 5c 7a c8 13 90 ...q......a.C.S-f..:.J....r\z...
0220 4b c0 3c 07 01 db 23 eb 3c f8 dc 31 7c 97 27 20 17 b3 1c 04 1c 2c c2 9c 87 1d 63 57 79 02 72 31 K.<...#.<..1|.'......,....cWy.r1
0240 3d 3c 50 03 7f 16 21 6a ec 05 fa 43 52 71 a5 77 43 73 31 25 3c 90 80 13 c5 88 eb e3 66 13 7d 0c =<P...!j...CRq.wCs1%<.......f.}.
0260 3c ba 04 8c 0b 7d ef 37 5f 40 2e 2b 01 0a e9 ff 7e bf 5d 15 48 45 64 b1 e1 4f 7f 25 09 f0 f0 04 <....}.7_@.+....~.].HEd..O.%....
0280 78 02 3c 3c 01 d5 4b 80 85 bc 22 8f d2 13 60 21 2f c8 07 4f 65 8e 84 07 81 ec 6a d5 78 49 2a 46 x.<<..K..."...`!/..Oe.....j.xI*F
02a0 30 d6 2e 75 0b 2e cf 26 3b 6e 56 2e 75 68 44 90 e6 94 b4 9b 0b 9f 3d 1d 37 6b 2a 05 01 dd 84 92 0..u...&;nV.uhD.......=.7k*.....
02c0 64 9b 61 1a d0 11 65 f5 92 94 84 c9 a1 e1 33 65 3c 74 7d c0 a4 ac ef 35 05 b2 d7 24 a8 cf 1a 1a d.a...e.......3e<t}....5...$....
02e0 88 f2 30 b1 11 26 ea 70 b0 75 6d d4 d5 6f 86 2b 94 3b ef 84 32 12 70 36 b4 ce ba 02 ef 9f 14 ba ..0..&.p.um..o.+.;..2.p6........
0300 bf a7 24 04 08 8e 85 c7 d2 70 67 d4 d5 d7 b8 73 ff e5 2c cc 77 86 d6 19 db 28 cd c8 e7 c6 8d d2 ..$......pg....s..,.w....(......
0320 8c f0 e1 51 50 0f 2f 89 11 de ef 20 25 72 e9 50 b0 bc c2 76 ef 17 07 fb 79 1d ac 71 ee 9e a8 df ...QP./.....%r.P...v....y..q....
0340 4b 45 80 ab fa b5 2c 29 dd 33 d6 c9 92 d2 5c 83 d5 15 26 e0 a8 63 6c 65 9e 87 a7 c5 a1 aa 8f 94 KE....,).3....\...&..cle........
0360 84 80 b8 d9 d7 06 3f 85 c5 55 b0 2d 25 3d 5d 48 3f cf ab 52 ac 5d 7a 32 06 5f 54 da 9d 4d c3 97 ......?..U.-%=]H?..R.]z2._T..M..
0380 ae 8d 4d 4a cf 49 d2 08 de 4f 9b dc 87 67 57 24 8d 30 92 57 99 94 56 c7 60 fd 30 46 fa 04 b0 47 ..MJ.I...O...gW$.0.W..V.`.0F...G
03a0 70 00 e8 4c c3 a9 f3 7d 41 82 da 34 d4 09 a6 09 e6 18 2c 10 5c 3d d2 83 e4 db 5f 19 b5 ad 64 83 p..L...}A..4......,.\=...._...d.
03c0 54 3f 11 8e 93 eb 89 9d 97 8e 2f 80 9f 0d 4e 67 7a 9b 66 0a 1e 06 9a 1c 7b d0 33 08 d7 e4 db 52 T?......../...Ngz.f.....{.3....R
03e0 53 30 01 19 9f fd 53 c1 13 a5 3e 99 e5 22 20 b3 a6 b5 82 97 8b 10 50 bc 9e 30 7b a1 a4 a9 88 2e S0....S...>.."........P..0{.....
0400 68 31 f8 ec 52 8a 3e 05 eb 80 9d 11 ff 66 5f 3f bc 52 f2 5c d0 33 66 67 13 66 4b d3 d0 6a f0 43 h1..R.>......f_?.R.\.3fg.fK..j.C
0420 11 d6 ff 3d d0 5b 49 02 32 8d 01 8f 03 6f 8f 21 90 ea 35 78 b3 0f 16 ac 32 eb 89 7c 18 0a cd 2c ...=.[I.2....o.!..5x....2..|...,
0440 24 a5 b9 82 25 82 b9 06 b3 05 33 80 06 83 09 59 3e f2 10 d0 63 d0 a7 20 a0 fb 16 38 3c 04 db 5a $...%.....3....Y>...c......8<..Z
0460 cd 0e a5 a4 2e b2 8a fe 06 e9 84 59 4d b9 54 d0 05 81 a1 74 43 6d e0 1a 2f 14 cc 01 1a 09 de 77 ...........YM.T....tCm../......w
0480 80 a0 6f f4 a4 e0 0f 0b de 71 d8 d5 0b 5b db cc ba 8a 26 8d 95 a8 b3 87 09 00 7a e3 66 75 54 21 ..o......q...[....&.......z.fuT!
04a0 2a 52 0f 70 a4 b6 bb a8 52 8c d6 9e fe 77 76 ae 44 81 6a 99 56 68 63 6e 28 26 a8 bd 0d 2e 0b 0d *R.p....R....wv.D.j.Vhcn(&......
04c0 77 54 2b 01 a3 49 40 bf 02 5f b8 3e 93 01 9c 72 0e 6e 8f 32 e1 6c 98 49 ae 04 1c f3 04 b8 71 dc wT+..I@.._.>...r.n.2.l.I......q.
04e0 71 c3 8a 88 22 b7 dc a1 92 f6 7b 02 dc ba fa 2b 87 0f bd 66 93 34 a6 b4 74 4a ba 5e f0 a2 23 3d q...".....{....+...f.4..tJ.^..#=
0500 b0 c3 13 e0 86 ab 95 7c ca 78 d8 f9 a1 54 50 6a 3a 29 2d 36 d8 93 e5 e2 fd 1b fe b7 98 ed ab 56 .......|.x...TPj:)-6...........V
0520 02 46 74 43 25 a9 1d f6 02 77 0f 17 0d 5a 90 0f fa 0d e8 3e df be a8 40 c7 4f 02 1a 2d 78 cb 66 .FtC%....w...Z.....>...@.O..-x.f
0540 a1 e0 c6 61 a4 ac 2d 61 b6 de 13 30 fc c9 9d 1f 0b 54 51 d1 ab 56 06 db 9a 61 71 35 37 fd 8e 1a ...a..-a...0.....TQ..V...aq57...
0560 07 34 9b 7d 03 2c a1 f8 75 db ed 83 90 a8 f6 8e eb bc 23 e1 4c c8 be 16 68 55 6e eb 7a 21 f8 d5 .4.}.,..u.........#.L...hUn.z!..
0580 60 5d 33 bc eb db dd c7 90 8a f8 40 6a a8 87 25 06 f7 65 72 fd 37 65 5a 4c 26 f3 5f 6d d7 80 41 `]3........@j..%..er.7eZL&._m..A
05a0 0b 6a 04 1d 04 65 c0 fd 06 3b 9a 61 b7 df f8 08 04 78 94 d9 06 78 78 02 2e 69 fc 03 1f 02 cf 3f .j...e...;.a.....x...xx..i.....?
05c0 61 3e 91 85 00 00 00 00 49 45 4e 44 ae 42 60 82 a>......IEND.B`.
311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350 351 352 353 354 355 356 357 358 359 360 361 362 363 364 365 366 367 368 369 370 371 372 373 374 375 376 377 378 379 380 381 382 383 384 385 386 387 388 389 390 391 392 393 394 395 396 397 398 399 400 401 402 403 404 405 406 407 408 409 410 411 412 413 414 415 416 417 418 419 420 421 422 423 424 425 426 427 428
/* ui-log.c: functions for log output
 *
 * Copyright (C) 2006 Lars Hjemli
 *
 * Licensed under GNU General Public License v2
 *   (see COPYING for full license text)
 */

#include "cgit.h"
#include "html.h"
#include "ui-shared.h"
#include "vector.h"

int files, add_lines, rem_lines;

/*
 * The list of available column colors in the commit graph.
 */
static const char *column_colors_html[] = {
	"<span class='column1'>",
	"<span class='column2'>",
	"<span class='column3'>",
	"<span class='column4'>",
	"<span class='column5'>",
	"<span class='column6'>",
	"</span>",
};

#define COLUMN_COLORS_HTML_MAX (ARRAY_SIZE(column_colors_html) - 1)

void count_lines(char *line, int size)
{
	if (size <= 0)
		return;

	if (line[0] == '+')
		add_lines++;

	else if (line[0] == '-')
		rem_lines++;
}

void inspect_files(struct diff_filepair *pair)
{
	unsigned long old_size = 0;
	unsigned long new_size = 0;
	int binary = 0;

	files++;
	if (ctx.repo->enable_log_linecount)
		cgit_diff_files(pair->one->sha1, pair->two->sha1, &old_size,
				&new_size, &binary, 0, ctx.qry.ignorews,
				count_lines);
}

void show_commit_decorations(struct commit *commit)
{
	struct name_decoration *deco;
	static char buf[1024];

	buf[sizeof(buf) - 1] = 0;
	deco = lookup_decoration(&name_decoration, &commit->object);
	while (deco) {
		if (!prefixcmp(deco->name, "refs/heads/")) {
			strncpy(buf, deco->name + 11, sizeof(buf) - 1);
			cgit_log_link(buf, NULL, "branch-deco", buf, NULL,
				      ctx.qry.vpath, 0, NULL, NULL,
				      ctx.qry.showmsg);
		}
		else if (!prefixcmp(deco->name, "tag: refs/tags/")) {
			strncpy(buf, deco->name + 15, sizeof(buf) - 1);
			cgit_tag_link(buf, NULL, "tag-deco", ctx.qry.head, buf);
		}
		else if (!prefixcmp(deco->name, "refs/tags/")) {
			strncpy(buf, deco->name + 10, sizeof(buf) - 1);
			cgit_tag_link(buf, NULL, "tag-deco", ctx.qry.head, buf);
		}
		else if (!prefixcmp(deco->name, "refs/remotes/")) {
			strncpy(buf, deco->name + 13, sizeof(buf) - 1);
			cgit_log_link(buf, NULL, "remote-deco", NULL,
				      sha1_to_hex(commit->object.sha1),
				      ctx.qry.vpath, 0, NULL, NULL,
				      ctx.qry.showmsg);
		}
		else {
			strncpy(buf, deco->name, sizeof(buf) - 1);
			cgit_commit_link(buf, NULL, "deco", ctx.qry.head,
					 sha1_to_hex(commit->object.sha1),
					 ctx.qry.vpath, 0);
		}
		deco = deco->next;
	}
}

void print_commit(struct commit *commit, struct rev_info *revs)
{
	struct commitinfo *info;
	char *tmp;
	int cols = revs->graph ? 3 : 2;
	struct strbuf graphbuf = STRBUF_INIT;
	struct strbuf msgbuf = STRBUF_INIT;

	if (ctx.repo->enable_log_filecount) {
		cols++;
		if (ctx.repo->enable_log_linecount)
			cols++;
	}

	if (revs->graph) {
		/* Advance graph until current commit */
		while (!graph_next_line(revs->graph, &graphbuf)) {
			/* Print graph segment in otherwise empty table row */
			html("<tr class='nohover'><td class='commitgraph'>");
			html(graphbuf.buf);
			htmlf("</td><td colspan='%d' /></tr>\n", cols);
			strbuf_setlen(&graphbuf, 0);
		}
		/* Current commit's graph segment is now ready in graphbuf */
	}

	info = cgit_parse_commit(commit);
	htmlf("<tr%s>", ctx.qry.showmsg ? " class='logheader'" : "");

	if (revs->graph) {
		/* Print graph segment for current commit */
		html("<td class='commitgraph'>");
		html(graphbuf.buf);
		html("</td>");
		strbuf_setlen(&graphbuf, 0);
	}
	else {
		html("<td>");
		tmp = fmt("id=%s", sha1_to_hex(commit->object.sha1));
		tmp = cgit_fileurl(ctx.repo->url, "commit", ctx.qry.vpath, tmp);
		html_link_open(tmp, NULL, NULL);
		cgit_print_age(commit->date, TM_WEEK * 2, FMT_SHORTDATE);
		html_link_close();
		html("</td>");
	}

	htmlf("<td%s>", ctx.qry.showmsg ? " class='logsubject'" : "");
	if (ctx.qry.showmsg) {
		/* line-wrap long commit subjects instead of truncating them */
		size_t subject_len = strlen(info->subject);

		if (subject_len > ctx.cfg.max_msg_len &&
		    ctx.cfg.max_msg_len >= 15) {
			/* symbol for signaling line-wrap (in PAGE_ENCODING) */
			const char wrap_symbol[] = { ' ', 0xE2, 0x86, 0xB5, 0 };
			int i = ctx.cfg.max_msg_len - strlen(wrap_symbol);

			/* Rewind i to preceding space character */
			while (i > 0 && !isspace(info->subject[i]))
				--i;
			if (!i) /* Oops, zero spaces. Reset i */
				i = ctx.cfg.max_msg_len - strlen(wrap_symbol);

			/* add remainder starting at i to msgbuf */
			strbuf_add(&msgbuf, info->subject + i, subject_len - i);
			strbuf_trim(&msgbuf);
			strbuf_add(&msgbuf, "\n\n", 2);

			/* Place wrap_symbol at position i in info->subject */
			strcpy(info->subject + i, wrap_symbol);
		}
	}
	cgit_commit_link(info->subject, NULL, NULL, ctx.qry.head,
			 sha1_to_hex(commit->object.sha1), ctx.qry.vpath, 0);
	show_commit_decorations(commit);
	html("</td><td>");
	html_txt(info->author);

	if (revs->graph) {
		html("</td><td>");
		tmp = fmt("id=%s", sha1_to_hex(commit->object.sha1));
		tmp = cgit_fileurl(ctx.repo->url, "commit", ctx.qry.vpath, tmp);
		html_link_open(tmp, NULL, NULL);
		cgit_print_age(commit->date, TM_WEEK * 2, FMT_SHORTDATE);
		html_link_close();
	}

	if (ctx.repo->enable_log_filecount) {
		files = 0;
		add_lines = 0;
		rem_lines = 0;
		cgit_diff_commit(commit, inspect_files, ctx.qry.vpath);
		html("</td><td>");
		htmlf("%d", files);
		if (ctx.repo->enable_log_linecount) {
			html("</td><td>");
			htmlf("-%d/+%d", rem_lines, add_lines);
		}
	}
	html("</td></tr>\n");

	if (revs->graph || ctx.qry.showmsg) { /* Print a second table row */
		html("<tr class='nohover'>");

		if (ctx.qry.showmsg) {
			/* Concatenate commit message + notes in msgbuf */
			if (info->msg && *(info->msg)) {
				strbuf_addstr(&msgbuf, info->msg);
				strbuf_addch(&msgbuf, '\n');
			}
			format_note(NULL, commit->object.sha1, &msgbuf,
			            PAGE_ENCODING,
			            NOTES_SHOW_HEADER | NOTES_INDENT);
			strbuf_addch(&msgbuf, '\n');
			strbuf_ltrim(&msgbuf);
		}

		if (revs->graph) {
			int lines = 0;

			/* Calculate graph padding */
			if (ctx.qry.showmsg) {
				/* Count #lines in commit message + notes */
				const char *p = msgbuf.buf;
				lines = 1;
				while ((p = strchr(p, '\n'))) {
					p++;
					lines++;
				}
			}

			/* Print graph padding */
			html("<td class='commitgraph'>");
			while (lines > 0 || !graph_is_commit_finished(revs->graph)) {
				if (graphbuf.len)
					html("\n");
				strbuf_setlen(&graphbuf, 0);
				graph_next_line(revs->graph, &graphbuf);
				html(graphbuf.buf);
				lines--;
			}
			html("</td>\n");
		}
		else
			html("<td/>"); /* Empty 'Age' column */

		/* Print msgbuf into remainder of table row */
		htmlf("<td colspan='%d'%s>\n", cols,
			ctx.qry.showmsg ? " class='logmsg'" : "");
		html_txt(msgbuf.buf);
		html("</td></tr>\n");
	}

	strbuf_release(&msgbuf);
	strbuf_release(&graphbuf);
	cgit_free_commitinfo(info);
}

static const char *disambiguate_ref(const char *ref)
{
	unsigned char sha1[20];
	const char *longref;

	longref = fmt("refs/heads/%s", ref);
	if (get_sha1(longref, sha1) == 0)
		return longref;

	return ref;
}

static char *next_token(char **src)
{
	char *result;

	if (!src || !*src)
		return NULL;
	while (isspace(**src))
		(*src)++;
	if (!**src)
		return NULL;
	result = *src;
	while (**src) {
		if (isspace(**src)) {
			**src = '\0';
			(*src)++;
			break;
		}
		(*src)++;
	}
	return result;
}

void cgit_print_log(const char *tip, int ofs, int cnt, char *grep, char *pattern,
		    char *path, int pager, int commit_graph)
{
	struct rev_info rev;
	struct commit *commit;
	struct vector vec = VECTOR_INIT(char *);
	int i, columns = 3;
	char *arg;

	/* First argv is NULL */
	vector_push(&vec, NULL, 0);

	if (!tip)
		tip = ctx.qry.head;
	tip = disambiguate_ref(tip);
	vector_push(&vec, &tip, 0);

	if (grep && pattern && *pattern) {
		pattern = xstrdup(pattern);
		if (!strcmp(grep, "grep") || !strcmp(grep, "author") ||
		    !strcmp(grep, "committer")) {
			arg = fmt("--%s=%s", grep, pattern);
			vector_push(&vec, &arg, 0);
		}
		if (!strcmp(grep, "range")) {
			/* Split the pattern at whitespace and add each token
			 * as a revision expression. Do not accept other
			 * rev-list options. Also, replace the previously
			 * pushed tip (it's no longer relevant).
			 */
			vec.count--;
			while ((arg = next_token(&pattern))) {
				if (*arg == '-') {
					fprintf(stderr, "Bad range expr: %s\n",
						arg);
					break;
				}
				vector_push(&vec, &arg, 0);
			}
		}
	}
	if (commit_graph) {
		static const char *graph_arg = "--graph";
		static const char *color_arg = "--color";
		vector_push(&vec, &graph_arg, 0);
		vector_push(&vec, &color_arg, 0);
		graph_set_column_colors(column_colors_html,
					COLUMN_COLORS_HTML_MAX);
	}

	if (path) {
		arg = "--";
		vector_push(&vec, &arg, 0);
		vector_push(&vec, &path, 0);
	}

	/* Make sure the vector is NULL-terminated */
	vector_push(&vec, NULL, 0);
	vec.count--;

	init_revisions(&rev, NULL);
	rev.abbrev = DEFAULT_ABBREV;
	rev.commit_format = CMIT_FMT_DEFAULT;
	rev.verbose_header = 1;
	rev.show_root_diff = 0;
	setup_revisions(vec.count, vec.data, &rev, NULL);
	load_ref_decorations(DECORATE_FULL_REFS);
	rev.show_decorations = 1;
	rev.grep_filter.regflags |= REG_ICASE;
	compile_grep_patterns(&rev.grep_filter);
	prepare_revision_walk(&rev);

	if (pager)
		html("<table class='list nowrap'>");

	html("<tr class='nohover'>");
	if (commit_graph)
		html("<th></th>");
	else
		html("<th class='left'>Age</th>");
	html("<th class='left'>Commit message");
	if (pager) {
		html(" (");
		cgit_log_link(ctx.qry.showmsg ? "Collapse" : "Expand", NULL,
			      NULL, ctx.qry.head, ctx.qry.sha1,
			      ctx.qry.vpath, ctx.qry.ofs, ctx.qry.grep,
			      ctx.qry.search, ctx.qry.showmsg ? 0 : 1);
		html(")");
	}
	html("</th><th class='left'>Author</th>");
	if (commit_graph)
		html("<th class='left'>Age</th>");
	if (ctx.repo->enable_log_filecount) {
		html("<th class='left'>Files</th>");
		columns++;
		if (ctx.repo->enable_log_linecount) {
			html("<th class='left'>Lines</th>");
			columns++;
		}
	}
	html("</tr>\n");

	if (ofs<0)
		ofs = 0;

	for (i = 0; i < ofs && (commit = get_revision(&rev)) != NULL; i++) {
		free(commit->buffer);
		commit->buffer = NULL;
		free_commit_list(commit->parents);
		commit->parents = NULL;
	}

	for (i = 0; i < cnt && (commit = get_revision(&rev)) != NULL; i++) {
		print_commit(commit, &rev);
		free(commit->buffer);
		commit->buffer = NULL;
		free_commit_list(commit->parents);
		commit->parents = NULL;
	}
	if (pager) {
		html("</table><div class='pager'>");
		if (ofs > 0) {
			cgit_log_link("[prev]", NULL, NULL, ctx.qry.head,
				      ctx.qry.sha1, ctx.qry.vpath,
				      ofs - cnt, ctx.qry.grep,
				      ctx.qry.search, ctx.qry.showmsg);
			html("&nbsp;");
		}
		if ((commit = get_revision(&rev)) != NULL) {
			cgit_log_link("[next]", NULL, NULL, ctx.qry.head,
				      ctx.qry.sha1, ctx.qry.vpath,
				      ofs + cnt, ctx.qry.grep,
				      ctx.qry.search, ctx.qry.showmsg);
		}
		html("</div>");
	} else if ((commit = get_revision(&rev)) != NULL) {
		html("<tr class='nohover'><td colspan='3'>");
		cgit_log_link("[...]", NULL, NULL, ctx.qry.head, NULL,
			      ctx.qry.vpath, 0, NULL, NULL, ctx.qry.showmsg);
		html("</td></tr>\n");
	}
}