]> git.cameronkatri.com Git - mandoc.git/commitdiff
Do not leak 64 bytes of heap memory every time a manual page calls
authorIngo Schwarze <schwarze@openbsd.org>
Mon, 4 Oct 2021 10:12:27 +0000 (10:12 +0000)
committerIngo Schwarze <schwarze@openbsd.org>
Mon, 4 Oct 2021 10:12:27 +0000 (10:12 +0000)
a user-defined macro.  Calls of standard mdoc(7) and man(7) macros
were unaffected, so the effect on OpenBSD manual pages was small,
about 80 Kilobytes grand total for a full run of "makewhatis
/usr/share/man".

Argument expansion contexts for user-defined macros are stored on
a stack that grows as needed if calls of user-defined macros are
nested or recursive.  Individual stack entries contain dynamically
allocated arrays of pointers to arguments; these argument arrays
also grow as needed if user-defined macros take more than eight
arguments.  The mistake was that argument arrays of already
initialized expansion contexts were leaked rather than reused on
subsequent macro calls.

I found this issue in a systematic hunt for memory leaks after
Michael <Stapelberg at Debian> reported memory exhaustion problems
on the production server manpages.debian.org.  This sub-Megabyte
leak is not the cause of Michael's trouble, though, where Gigabytes
of memory are being wasted.  We are still investigating whether the
original problem may be related to his supervisor process, which is
written in Go, rather than to mandoc.

roff.c

diff --git a/roff.c b/roff.c
index de75a260f5a769a571c5b077e14aea8ebe590b0f..2d85e39fad56f60a0ef6f3d7d87738d934176a5a 100644 (file)
--- a/roff.c
+++ b/roff.c
@@ -1,4 +1,4 @@
-/* $Id: roff.c,v 1.378 2021/08/10 12:55:04 schwarze Exp $ */
+/* $Id: roff.c,v 1.379 2021/10/04 10:12:27 schwarze Exp $ */
 /*
  * Copyright (c) 2010-2015, 2017-2020 Ingo Schwarze <schwarze@openbsd.org>
  * Copyright (c) 2008-2012, 2014 Kristaps Dzonsons <kristaps@bsd.lv>
@@ -3949,9 +3949,7 @@ roff_userdef(ROFF_ARGS)
                r->mstacksz += 8;
        }
        ctx = r->mstack + r->mstackpos;
-       ctx->argsz = 0;
        ctx->argc = 0;
-       ctx->argv = NULL;
 
        /*
         * Collect pointers to macro argument strings,