* bug#28784: 'guix publish' leaks memory
@ 2017-10-11 11:45 Ludovic Courtès
2017-10-11 13:22 ` Ludovic Courtès
2017-10-11 19:41 ` bug#28784: Simplifying revealed ports Ludovic Courtès
0 siblings, 2 replies; 3+ messages in thread
From: Ludovic Courtès @ 2017-10-11 11:45 UTC (permalink / raw)
To: 28784
On the build farms, memory usage of the ‘guix publish’ process increases
slowly but indefinitely, it seems.
This can be reproduced by starting ‘guix publish --cache /tmp/cache -C’
and launching ‘guix weather’, which queries many narinfos, thereby
triggering the compression of many store items to /tmp/cache.
Ludo’.
^ permalink raw reply [flat|nested] 3+ messages in thread
* bug#28784: 'guix publish' leaks memory
2017-10-11 11:45 bug#28784: 'guix publish' leaks memory Ludovic Courtès
@ 2017-10-11 13:22 ` Ludovic Courtès
2017-10-11 19:41 ` bug#28784: Simplifying revealed ports Ludovic Courtès
1 sibling, 0 replies; 3+ messages in thread
From: Ludovic Courtès @ 2017-10-11 13:22 UTC (permalink / raw)
To: 28784-done
ludo@gnu.org (Ludovic Courtès) skribis:
> On the build farms, memory usage of the ‘guix publish’ process increases
> slowly but indefinitely, it seems.
This program reproduced the problem:
--8<---------------cut here---------------start------------->8---
(use-modules (guix zlib)
(ice-9 format)
(rnrs io ports))
(define (display-heap-size)
(format #t "heap size: ~,2h MiB~%"
(/ (assoc-ref (gc-stats) 'heap-size) (expt 2. 20))))
(let loop ((i 0))
(when (zero? (modulo i 1000))
(display-heap-size))
(let ((port (open-file "/dev/null" "w0")))
(call-with-gzip-output-port port
(lambda (port)
(display (make-string 1000 #\a) port))))
(loop (+ 1 i)))
--8<---------------cut here---------------end--------------->8---
This is fixed in 85a2b58987bc32e33e63bea86c1a94496b796ae9.
Ludo’.
^ permalink raw reply [flat|nested] 3+ messages in thread
* bug#28784: Simplifying revealed ports
2017-10-11 11:45 bug#28784: 'guix publish' leaks memory Ludovic Courtès
2017-10-11 13:22 ` Ludovic Courtès
@ 2017-10-11 19:41 ` Ludovic Courtès
1 sibling, 0 replies; 3+ messages in thread
From: Ludovic Courtès @ 2017-10-11 19:41 UTC (permalink / raw)
To: guile-devel; +Cc: 28784
[-- Attachment #1: Type: text/plain, Size: 745 bytes --]
Hello,
While debugging <https://bugs.gnu.org/28784>, I was surprised to see
that revealed ports were not GC’d at all. The manual suggests otherwise
(info "(guile) Ports and File Descriptors"):
If a port’s revealed count is greater than zero, the file descriptor
will not be closed when the port is garbage collected.
That revealed ports are not GC’d makes it easy to shoot oneself in the
foot the way I did (huge memory leak).
So what about the simplification below?
It creates one observable difference: ‘close-port’ does nothing on a
revealed port, whereas until now it would close the port even if it has
a non-zero revealed count. I would argue that it’s acceptable though.
Thoughts?
Ludo’.
[-- Warning: decoded text below may be mangled, UTF-8 assumed --]
[-- Attachment #2: the patch --]
[-- Type: text/x-patch, Size: 3785 bytes --]
diff --git a/libguile/fports.c b/libguile/fports.c
index 94092b872..ee6ac0bf1 100644
--- a/libguile/fports.c
+++ b/libguile/fports.c
@@ -1,6 +1,6 @@
/* Copyright (C) 1995, 1996, 1997, 1998, 1999, 2000, 2001, 2002, 2003,
* 2004, 2006, 2007, 2008, 2009, 2010, 2011, 2012, 2013,
- * 2014, 2015 Free Software Foundation, Inc.
+ * 2014, 2015, 2017 Free Software Foundation, Inc.
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public License
@@ -467,8 +467,6 @@ fport_input_waiting (SCM port)
#define SCM_REVEALED(x) (SCM_FSTREAM(x)->revealed)
-static SCM revealed_ports = SCM_EOL;
-static scm_i_pthread_mutex_t revealed_lock = SCM_I_PTHREAD_MUTEX_INITIALIZER;
/* Find a port in the table and return its revealed count.
Also used by the garbage collector.
@@ -476,13 +474,7 @@ static scm_i_pthread_mutex_t revealed_lock = SCM_I_PTHREAD_MUTEX_INITIALIZER;
int
scm_revealed_count (SCM port)
{
- int ret;
-
- scm_i_pthread_mutex_lock (&revealed_lock);
- ret = SCM_REVEALED (port);
- scm_i_pthread_mutex_unlock (&revealed_lock);
-
- return ret;
+ return SCM_REVEALED (port);
}
SCM_DEFINE (scm_port_revealed, "port-revealed", 1, 0, 0,
@@ -503,25 +495,14 @@ SCM_DEFINE (scm_set_port_revealed_x, "set-port-revealed!", 2, 0, 0,
"The return value is unspecified.")
#define FUNC_NAME s_scm_set_port_revealed_x
{
- int r, prev;
+ int r;
port = SCM_COERCE_OUTPORT (port);
SCM_VALIDATE_OPFPORT (1, port);
r = scm_to_int (rcount);
-
- scm_i_pthread_mutex_lock (&revealed_lock);
-
- prev = SCM_REVEALED (port);
SCM_REVEALED (port) = r;
- if (r && !prev)
- revealed_ports = scm_cons (port, revealed_ports);
- else if (prev && !r)
- revealed_ports = scm_delq_x (port, revealed_ports);
-
- scm_i_pthread_mutex_unlock (&revealed_lock);
-
return SCM_UNSPECIFIED;
}
#undef FUNC_NAME
@@ -539,18 +520,7 @@ SCM_DEFINE (scm_adjust_port_revealed_x, "adjust-port-revealed!", 2, 0, 0,
SCM_VALIDATE_OPFPORT (1, port);
a = scm_to_int (addend);
- if (!a)
- return SCM_UNSPECIFIED;
-
- scm_i_pthread_mutex_lock (&revealed_lock);
-
SCM_REVEALED (port) += a;
- if (SCM_REVEALED (port) == a)
- revealed_ports = scm_cons (port, revealed_ports);
- else if (!SCM_REVEALED (port))
- revealed_ports = scm_delq_x (port, revealed_ports);
-
- scm_i_pthread_mutex_unlock (&revealed_lock);
return SCM_UNSPECIFIED;
}
@@ -668,6 +638,11 @@ fport_close (SCM port)
{
scm_t_fport *fp = SCM_FSTREAM (port);
+ if (SCM_REVEALED (port) > 0)
+ /* The port has a non-zero revealed count, so don't close the
+ underlying file descriptor. */
+ return;
+
scm_run_fdes_finalizers (fp->fdes);
if (close (fp->fdes) != 0)
/* It's not useful to retry after EINTR, as the file descriptor is
diff --git a/libguile/fports.h b/libguile/fports.h
index afb8ba771..e397fcc59 100644
--- a/libguile/fports.h
+++ b/libguile/fports.h
@@ -3,7 +3,8 @@
#ifndef SCM_FPORTS_H
#define SCM_FPORTS_H
-/* Copyright (C) 1995,1996,1997,1998,1999,2000,2001, 2006, 2008, 2009, 2011, 2012 Free Software Foundation, Inc.
+/* Copyright (C) 1995-2001, 2006, 2008, 2009, 2011, 2012,
+ * 2017 Free Software Foundation, Inc.
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public License
@@ -33,9 +34,8 @@
typedef struct scm_t_fport {
/* The file descriptor. */
int fdes;
- /* Revealed count; 0 indicates not revealed, > 1 revealed. Revealed
- ports do not get garbage-collected. */
- int revealed;
+ /* Revealed count; 0 indicates not revealed, > 1 revealed. */
+ unsigned int revealed;
/* Set of scm_fport_option flags. */
unsigned options;
} scm_t_fport;
^ permalink raw reply related [flat|nested] 3+ messages in thread
end of thread, other threads:[~2017-10-11 19:42 UTC | newest]
Thread overview: 3+ messages (download: mbox.gz follow: Atom feed
-- links below jump to the message on this page --
2017-10-11 11:45 bug#28784: 'guix publish' leaks memory Ludovic Courtès
2017-10-11 13:22 ` Ludovic Courtès
2017-10-11 19:41 ` bug#28784: Simplifying revealed ports Ludovic Courtès
Code repositories for project(s) associated with this external index
https://git.savannah.gnu.org/cgit/guix.git
This is an external index of several public inboxes,
see mirroring instructions on how to clone and mirror
all data and code used by this external index.