Comment by cogman10

5 hours ago

> very exotic requirements

I'd be interested to know what you are thinking.

The primary exotic thing I can imagine is an architecture lacking the ability to do atomic operations. But even in that case, C11 has atomic operations [1] built in. So worst case, the C library for the target architecture would likely boil down to mutex operations.

[1] https://en.cppreference.com/w/c/atomic.html

Well, basically, yeah, if your platform lacks support for atomics, or if you'd need some extra functionality around the shared pointer like e.g. logging the shared pointer refcounts while enforcing consistent ordering of logs (which can be useful if you're unfortunate enough to have to debug a race condition where you need to pay attention to refcounts, assuming the extra mutex won't make your heisenbug disappear), or synchronizing something else along with the refcount (basically a "fat", custom shared pointer that does more than just shared-pointering).

  • Does there exist any platform which has multithreading but not atomics? Such a platform would be quite impractical as you can't really implement locks or any other threading primitive without atomics.

    • > Does there exist any platform which has multithreading but not atomics?

      Yes. Also, almost every platform I know that supports multi threading and atomics doesn’t support atomics between /all/ possible masters. Consider a microcontroller with, say, two Arm cores (multithreaded, atomic-supporting) and a DMA engine.

    • Certainly such systems can pretty readily exist. You merely need atomic reads/writes in order to implement locks.

      You can't create userspace locks which is a bummer, but the OS has the capability of enforcing locks. That's basically how early locking worked.

      The main thing needed to make a correct lock is interrupt protection. Something every OS has.

      To go fast, you need atomic operations. It especially becomes important if you are dealing with multiple cores. However, for a single core system atomics aren't needed for the OS to create locks.

      2 replies →

Which platforms might that be? Even MIPS has atomics (at least pointer sized last i checked).

  • AFIAK, and I'm not MIPS expert, but I believe it doesn't have the ability to add a value directly to a memory address. You have to do something like

        // Not real MIPS, just what I've gleaned from a brief look at some docs
        LOAD addr, register
        ADD 1, register
        STORE register, addr
    

    The LOAD and STORE are atomic, but the `ADD` happens out of band.

    That's a problem if any sort of interrupt happens (if you are multi-threading then a possibility). If it happens at the load, then a separate thread can update "addr" which mean the later STORE will stomp on what's there.

    x86 and ARM can do

        ADD 1, addr
    

    as well as other instructions like "compare and swap"

        LOAD addr, register
        MOV register, register2
        ADD 1, register2
        COMPARE_AND_SWAP addr, register, register2
        if (cas_failed) { try again }

    • On MIPS you can simulate atomics with a load-linked/store-conditional (LL/SC) loop. If another processor has changed the same address between the LL and SC instructions, the SC fails to store the result and you have to retry. The underlying idea is that the processors would have to communicate memory accesses to each other via the cache coherence protocol anyway, so they can easily detect conflicting writes between the LL and SC instructions. It gets more complicated with out-of-order execution...

          loop: LL r2, (r1)
                ADD r3, r2, 1
                SC r3, (r1)
                BEQ r3, 0, loop
                NOP