This wiki article contains incomplete and informal notes about the MPS, the precursor to more formal documentation. Not confidential. Readership: MPS developers.

These notes on Allocation Points were written by RHSK between 2006-06-12 and 2006-06-23, following research and discussion with RB and NB.

Warning: this article is more of a technical discussion than a set of facts. It is not authoritative. This is not (yet) an accurate statement of the MPS ap protocol. RHSK 2006-06-23.

Synchronization

The synchronization issues are a little tricky.

scenario

The MPS allocation point protocol is a binary protocol. You don't have to use the mps_reserve and mps_commit macros, but they are conventional and useful pre-packaged implementations. I'm using them as an example of correct protocol, referring to mps.h#17. It looks like this:

#define mps_reserve(_p_o, _mps_ap, _size) \
  ((char *)(_mps_ap)->alloc + (_size) > (char *)(_mps_ap)->alloc && \
   (char *)(_mps_ap)->alloc + (_size) <= (char *)(_mps_ap)->limit ? \
     ((_mps_ap)->alloc = \
       (mps_addr_t)((char *)(_mps_ap)->alloc + (_size)), \
      *(_p_o) = (_mps_ap)->init, \
      MPS_RES_OK) : \
     mps_ap_fill(_p_o, _mps_ap, _size))

#define mps_commit(_mps_ap, _p, _size) \
  ((_mps_ap)->init = (_mps_ap)->alloc, \
   (_mps_ap)->limit != 0 || mps_ap_trip(_mps_ap, _p, _size))

Abstractly, the mutator code has a cycle of four operations on the allocation point:

Lr
read Limit at reserve
A
write Alloc at reserve
I
write Init at commit
Lc
read Limit at commit

The normal cycle is Lr A I Lc.

After Lr, the mutator checks that A + size <= Lr. If this fails, then the mutator does mps_ap_fill instead of A. So Lr mps_ap_fill I Lc is also a valid cycle.

After Lc, the mutator checks that Lc != 0. If this fails, the mutator will call mps_ap_trip before beginning a new cycle. So Lr A I Lc mps_ap_trip is also a valid cycle.

The mutator cannot interrupt the collector.

The collector can interrupt the mutator (between any two mutator instructions). The collector can only see the A, I, and mps_ap_trip (if present) operations; Lr and Lc are invisible to it.

If the ap is not going to trip, the collector sees two equivalence classes:

If the ap is going to trip, the collector sees four equivalence classes:

The collector has two responsibilities:

FIX
to fix (or not fix) references TO the object;
SCAN
to scan (or not scan) references IN the object.

A mutator with ambiguous references only (to the new object)

Let's see what's required for a mutator that promises to make no exact references to the new object until Lc succeeds, and promises to have at least one ambiguous reference to the new object when it sets I.

Consider a flip -- f1 -- in state A.

The collector at f1 sees (A), and recognises that the I..A region is unmanaged memory. The object is never going to be valid, because it may contain stale references from before the flip, and the collector can't fix them.

The f1 collector writes L = 0, tripping the allocation point, and the rest of this collection is easy. The FIX responsibility is easy: all references to the new object are ambiguous, so the collector can fix them or not -- it doesn't matter. For SCAN, the collector must not scan the new object (because it is not yet formatted), so it must not scan the buffered pool beyond I.

Further collections at this point are idempotent.

Now, suppose the mutator then restarts, but only gets as far as (I Lc) before we have a second collection and a second flip -- f2.

For FIX, f2 can fix or not -- it still doesn't matter.

But for SCAN, f2 MUST NOT scan the new object. Why not?

The mutator's format code will be okay: the fact that I is set tells f2 that the object is fully formatted.

But there is a problem: the object may contain stale exact references -- unfixed references from before f1. These stale exact references will be bogus. Opinions differ on the implications of bogus exact references. See the next section for a discussion.

Additionally, the collector is not obliged to scan the object: this object is dead! The mutator will later notice that Lr == 0 and realise the object is invalid. The mutator is not relying on this object to keep anything else alive.

It is easy to avoid scanning the invalid object: the first flip in state (A), f1, must record the value of I. Call the recorded pointer If1. Collections must not scan beyond If1. If1 remains set until the mutator calls mps_ap_trip, which resets If1 to 0. [Note: I have not checked whether the MPS records and uses If1 -- RHSK 2006-06-12]

How bad is a bogus exact reference?

Tucker said:

I believe the collector must be prepared to deal with exact dangling pointers anyways, so this complex mechanism is unnecessary. [//info.ravenbrook.com/project/mps/mail/1997/05/12/12-46/1.txt:X-MMInfo-Tag: mail.ptw.1997-05-12.12-46(1)]

I'm not convinced.

Fixing a bogus exact reference is bad news. The worst case is spoofing: Imagine the bogus exact reference points at part of a JPEG that just happens to look like a validly formatted object. If this JPEG is in the middle of an object in a moving MPS pool, fixing the bogus exact reference will overwrite a portion of the JPEG with a forwarding object. It gets worse: the spoof object is preserved and will later be scanned, producing more bogus exact references! For "JPEG", you can substitute "encrypted string".

The only safe way to cope with such a bogus reference is for Fix to verify that the reference points to the start of a real object, which it could do by iterating the format skip method from some known point in the pool.

The consequences of fixing a bogus exact reference may be:

Also, more arcanely, I wonder whether there's a problem if the referent violates the tri-colour invariant (when scanning at black)? Does the MPS care about this, or does it 'safely' just cause more conservatism? Does MPS detect and flag this in debug versions? Could it?

If MPS or format code detects a bogus exact reference, it could assert, and/or to fix them to a canonical special value.

Some possible contracts between mutator and collector:

  1. Mutator and collector promise never to have bogus exact references. The collector is permitted to crash if it encounters one.
  2. Mutator agrees that bogus exact references are wrong. But mutator would like a helpful error message, please. (Collector is not permitted to crash).
  3. Mutator wants permission to keep bogus exact references in certain circumstances, such as on the stack. Collector must cope safely. Need to specify how this is guaranteed, based on intimate knowledge of the mutator's behaviour. Dodgy.
  4. Mutator wants permission to keep bogus exact references in general. Collector must safely cope with all the cases of bogus exact references listed above. I believe the MPS does not currently do this. It would require a very fast check at fix time that the reference is to the start of a valid object in an MPS client pool.

with exact refs

For an all-exact mode of use, mutator must make an exact pointer before setting I=A, and must keep it at least until it after it has read Lc to see whther the commit succeeded.

So, some additions to the AP User's Guide, in the section on mps_reserve:

Secondly, "raw data" means that any references TO the new object are treated like other references to unmanaged memory. [.belief.refs-to-uninit-safe: We're 'sure', but I need to check this. What does Fix actually do with a pointer into the init-alloc zone? We hope it ignores it. RHSK 2006-06-09] Because of this, you are permitted to connect the new object into your graph of managed objects immediately. The MPS gives you these guarantees:

And to the section on mps_commit:

When this happens the client should take care to clear up any managed references to the (now vanished) new object.

[But there's a hole here, before the client does this. Are managed (aka scanned) references TO it still safe? They were safe during building (by .belief.refs-to-uninit-safe). But now the AP pointers have gone away. Are they still safe? Clearly, if they are only RankAMBIG, they are safe. What if they are RankEXACT? RHSK 2006-06-09]

[Discussion with RB 2006-06-09: yes, that's a problem for exact references. Must not make any exact refs to a new object. And unmanaged refs are not sufficient, because they won't preserve the new object during commit. So must make at least one ambiguous ref to new object before calling commit. That's the truth currently. There are various ways to solve this to allow purely-exact mutators. For instance, keep the old init..alloc address-space flagged as a zombie zone, until some communication with mutator (perhaps another reserve from same AP?) indicates that mutator has removed all those pesky exact refs to the now-dead ex-new object. RHSK 2006-06-09]

See issues with unmanaged workspace for a discussion of the "abort" phase, which we need to define for allocation points.

The MPS needs to keep the buffer mapped for some time -- until the mutator has stopped writing into it. The MPS needs to remember If1 forever. The MPS needs to remember the rest of the about-to-be-detached buffer structure for some time -- when is the MPS permitted to detach the buffer?

Keeping the buffer mapped consumes resource. There must be a contract with the mutator about when the MPS is permitted to free it.

We could define the protocol such that there is a window: after the mutator discovers that it has lost the race, but while the collector is still remembering that this object is invalid. The mutator could use this window to null-out exact references to the new object.

This window could be after Lc and before calling mps_ap_trip. During this time, the collector must keep the new object as pinned unmanaged memory. By calling mps_ap_trip, the mutator promises that it has nulled-out any exact references to the new object, and the collector in mps_ap_trip may unmap the memory or safely re-use it without fear of spoofing.

Or, we could leave the window open until mps_ap_fill is called. We just need to define it.

flipping at other times

Perhaps the collector wants to flip even when the ap is not at (A). Need to analyse synchronization issues here.

B. Document History

(As part of GC article)
  2006-06-12  RHSK  Allocation point internals: Scenario; A mutator with 
                    ambiguous references only (to the new object); How 
                    bad is a bogus exact reference?; quick notes on: with 
                    exact refs; flipping at other times.
  2006-06-15  RHSK  Allocation Points: clarifications and further notes 
                    into User Guide and Internals, including initializing 
                    as-yet unused parts of new objects to reduce spoofing.
  2006-06-22  RHSK  Move rough notes .talk.RB.2006-06-13 etc elsewhere.
                    Make AP User Guide consistently .assume.ambig-workspace,
                    moving notes to APInternals; add Intro and Mistakes;
                    check example code.

(As part of AP User Guide article)
  2006-06-22  RHSK  Created from GC article.

(This article)
  2006-06-23  RHSK  Created from AP User Guide article.

C. Copyright and License

This document is copyright © 2006 Ravenbrook Limited. All rights reserved. This is an open source license. Contact Ravenbrook for commercial licensing options.

Redistribution and use in source and binary forms, with or without modification, are permitted provided that the following conditions are met:

  1. Redistributions of source code must retain the above copyright notice, this list of conditions and the following disclaimer.
  2. Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the following disclaimer in the documentation and/or other materials provided with the distribution.
  3. Redistributions in any form must be accompanied by information on how to obtain complete source code for the this software and any accompanying software that uses this software. The source code must either be included in the distribution or be available for no more than the cost of distribution plus a nominal fee, and must be freely redistributable under reasonable conditions. For an executable file, complete source code means the source code for all modules it contains. It does not include source code for modules or files that typically accompany the major components of the operating system on which the executable file runs.

This software is provided by the copyright holders and contributors "as is" and any express or implied warranties, including, but not limited to, the implied warranties of merchantability, fitness for a particular purpose, or non-infringement, are disclaimed. In no event shall the copyright holders and contributors be liable for any direct, indirect, incidental, special, exemplary, or consequential damages (including, but not limited to, procurement of substitute goods or services; loss of use, data, or profits; or business interruption) however caused and on any theory of liability, whether in contract, strict liability, or tort (including negligence or otherwise) arising in any way out of the use of this software, even if advised of the possibility of such damage.