diff options
author | Mattias EngdegÄrd <mattiase@acm.org> | 2022-03-26 16:44:18 +0100 |
---|---|---|
committer | Mattias EngdegÄrd <mattiase@acm.org> | 2022-03-26 17:11:40 +0100 |
commit | 2dfeea8962751718168494c0560d69e678794b39 (patch) | |
tree | feb733329b75e74f0e3569706a0d17f3dcc79022 /src/lread.c | |
parent | e96061de95d053a4c5e303c7f75e0e928e474938 (diff) | |
download | emacs-2dfeea8962751718168494c0560d69e678794b39.tar.gz emacs-2dfeea8962751718168494c0560d69e678794b39.tar.bz2 emacs-2dfeea8962751718168494c0560d69e678794b39.zip |
Fix reader infinite recursion for circular mixed-type values
Make sure that the value added to the `read_objects_completed` set is
the one we actually return; previously this wasn't the case for conses
because of an optimisation (bug#54501).
Also add a check for vacuous self-references such as #1=#1# instead of
returning a nonsense value from thin air.
* src/lread.c (read1): Treat numbered conses correctly as described
above. Detect vacuous self-references.
* test/src/lread-tests.el (lread-test-read-and-print)
(lread-test-circle-cases, lread-circle): Add tests.
Diffstat (limited to 'src/lread.c')
-rw-r--r-- | src/lread.c | 46 |
1 files changed, 30 insertions, 16 deletions
diff --git a/src/lread.c b/src/lread.c index 6130300b0a2..2538851bac6 100644 --- a/src/lread.c +++ b/src/lread.c @@ -3488,6 +3488,29 @@ read1 (Lisp_Object readcharfun, int *pch, bool first_in_list, bool locate_syms) /* Read the object itself. */ Lisp_Object tem = read0 (readcharfun, locate_syms); + if (CONSP (tem)) + { + if (BASE_EQ (tem, placeholder)) + /* Catch silly games like #1=#1# */ + invalid_syntax ("nonsensical self-reference", + readcharfun); + + /* Optimisation: since the placeholder is already + a cons, repurpose it as the actual value. + This allows us to skip the substition below, + since the placeholder is already referenced + inside TEM at the appropriate places. */ + Fsetcar (placeholder, XCAR (tem)); + Fsetcdr (placeholder, XCDR (tem)); + + struct Lisp_Hash_Table *h2 + = XHASH_TABLE (read_objects_completed); + ptrdiff_t i = hash_lookup (h2, placeholder, &hash); + eassert (i < 0); + hash_put (h2, placeholder, Qnil, hash); + return placeholder; + } + /* If it can be recursive, remember it for future substitutions. */ if (! SYMBOLP (tem) @@ -3502,24 +3525,15 @@ read1 (Lisp_Object readcharfun, int *pch, bool first_in_list, bool locate_syms) } /* Now put it everywhere the placeholder was... */ - if (CONSP (tem)) - { - Fsetcar (placeholder, XCAR (tem)); - Fsetcdr (placeholder, XCDR (tem)); - return placeholder; - } - else - { - Flread__substitute_object_in_subtree - (tem, placeholder, read_objects_completed); + Flread__substitute_object_in_subtree + (tem, placeholder, read_objects_completed); - /* ...and #n# will use the real value from now on. */ - i = hash_lookup (h, number, &hash); - eassert (i >= 0); - set_hash_value_slot (h, i, tem); + /* ...and #n# will use the real value from now on. */ + i = hash_lookup (h, number, &hash); + eassert (i >= 0); + set_hash_value_slot (h, i, tem); - return tem; - } + return tem; } /* #n# returns a previously read object. */ |