Per, John, Shiro, thank you very much for your prompt feedback!

Let me address the points you have made:

(1) Procedure properties are related to this proposal (and one can be implemented with the other, and vice versa) but are not equivalent:

(a) Procedure properties (at least those defined by Guile) are not encapsulated because any piece of code can simply replace the list.
(b) Procedure properties affect all procedures, while this proposal allows a compiler to keeps its legacy representation for all non-tagged procedures (the default).
(c) Procedure properties are somewhat of a higher level.  When viewed as tagged procedures in the sense of this proposal, their tag is always an alist (at least conceptually).  So one can view this proposal as the low-level API, on which things like procedure properties or hooks (see document) can be built.
(d) At least on R6RS systems, ordinary procedures need not have an identity (with respect to "eq?" and "eqv?") allowing further optimizations.  On such systems, procedure properties for all procedures are not implementable.
(e) As tags are immutable, the compiler can do more optimizations than with the highly dynamic properties.

(2) In any case, procedure properties and this proposal are not exclusive.

(3) Thanks for mentioning the Chicken interface, which allows an efficient implementation of this proposal in Chicken.  I'll mention this and Kawa's and Guile's procedure properties in the next draft.

(4) I've thought long about whether extending procedures with fields/a tag or making records callable.  I finally chose the former because finding a common approach for all record systems in use would have been nigh impossible.  In most actual code, the tag field will be initialized to a record that holds the actual fields, so the expressiveness is eventually the same.

(5) The reason why the tag is immutable is two-fold:  Firstly, exactly because it enables lambda-lifting of so-tagged procedures and leaves the closures of top-level procedures constant.  The second reason is encapsulation.  If some library creates tagged procedures whose tag holds an opaque record, external code cannot interfere with the library by overwriting the tag.  If one needs mutability, one can always use a record (or box or vector) with a mutable field as a tag.

(6) I concur with Shiro on the (non-)generativeness of the lambda form.  That's an important property (or, rather, non-property).  This API was designed to make an efficient implementation possible and simple without necessitating any performance regressions for existing procedures.  Moreover, it was designed to make more specialized APIs (like those dealing with procedure property lists, MIT/GNU Scheme application hooks, SRFI 173 hooks, procedures with mutable tags, callable record types, etc.) efficiently and easily implementable using this SRFI.

Marc

Am Mi., 1. Sept. 2021 um 04:49 Uhr schrieb Shiro Kawai <xxxxxx@gmail.com>:
On Tue, Aug 31, 2021 at 3:57 PM Per Bothner <xxxxxx@bothner.com> wrote:

I think an intuitive semantics is that each evaluation of (lambda ...)
conceptually creates a district procedure value - just like (cons ...) does.
If a compiler can determine that no-one modifies a procedure (either though
set-procedure-property! *or* by modifying closed mutable values) *and* no-one
compares the procedures using eq? then a compiler can merge them to a single
value - just as it can for pairs created by cons. 

What real optimization would be lost under this semantics - which I suspect
matches most real implementations?

Lifting invariant lambdas is very effective optimization, and can be done even if the resulting procedure escapes.
The cases of a compiler being able to determine the listed conditions is a small subset of the actual cases, unless you do the whole program analysis.

One of the common occurrences is at the default value of an optional argument expecting a procedure.

   (define (foo xs :optional (proc (lambda (x) ....)))
       .... (find proc xs) ...)

It uses Gauche's optional argument syntax, but you can get the idea.   Allocating new closure every time when foo is called without the optional argument is a huge performance hit, but the proc itself escapes and you won't know if its property is mutated there.

Note that 'cons' is specified to return a freshly allocated pair, but there's no such requirements in 'lambda'.

I once implemented mutable procedure properties in Gauche but was bitten by unpredictable behaviors, which turned out to be the interference of lambda lifting, so I abandoned the idea.  (We still have internal properties, but not exposed to the public, and mutation is restricted for internal use that are determined to be safe).