rom1v 1 day ago

Related to the discussion: "A fork() in the road": https://www.microsoft.com/en-us/research/wp-content/uploads/...

> ABSTRACT

> The received wisdom suggests that Unix’s unusual combination of fork() and exec() for process creation was an inspired design. In this paper, we argue that fork was a clever hack for machines and programs of the 1970s that has long outlived its usefulness and is now a liability. We catalog the ways in which fork is a terrible abstraction for the modern programmer to use, describe how it compromises OS implementations, and propose alternatives.

> As the designers and implementers of operating systems, we should acknowledge that fork’s continued existence as a first-class OS primitive holds back systems research, and deprecate it. As educators, we should teach fork as a historical artifact, and not the first process creation mechanism students encounter.

  • pizlonator 1 day ago

    Fork is marvelous for the zygote pattern

    Hard to come up with an optimization that is equally efficient and elegant

    • toast0 1 day ago

      The zygote pattern[1] is a great optimization to deal with the cost of forking, but IMHO, being able to inexpensively spawn a carefully tailored process regardless of the size and scope of the current process would be better.

      I would guess it would be a small difference in measurable performance between zygote and a direct clean spawn, but it's one less trick an application needs to do, and it would be very helpful for libraries that spawn things. Spawning inside a library isn't always a great thing to do, but some things would really benefit from process level isolation.

      [1] In case one isn't aware, the zygote pattern involves forking a 'zygote' process during application startup, and having that process do any forks that need to happen during application runtime. This reduces the cost of forking in large applications, because the zygote will have few fds open and use little memory. This lets your large application spawn new processes without delaying the application or the startup of the new processes. Some applications will spawn many zygotes to allow parallelism for spawning at runtime.

      • pizlonator 1 day ago

        You're referring to something else, and maybe I'm using the term "zygote" incorrectly.

        In all uses of zygotes that I have seen, here's what's really happening:

        - `fork` is being used to reduce the cost of starting a process that has a high start-up cost. So, you start one process, run it through the expensive initialization, and then fork it from there to start new processes.

        - To make this even faster, you have a pool of pre-forked processes sit around.

        - Having pre-forked processes sitting around ready to be used is not expensive because of the CoW property and the fact that a process that forks and then immediately pauses will not have triggered any significant CoW yet.

        So, the zygote optimization you speak of is in practice only meaningful on top of systems that are using an optimization uniquely enabled by `fork` (avoiding process initialization costs by cloning a process), and that zygote optimization is further optimized by another property of `fork` (memory sharing of forked processes that haven't done anything else yet).

        • toast0 1 day ago

          Oh I see. I guess your zygotes have developed more than mine. I think Google may have coined or at least popularized the term zygote for this in Chrome and Android, Chrome documentation [1] says:

          > A zygote process is one that listens for spawn requests from a main process and forks itself in response. Generally they are used because forking a process after some expensive setup has been performed can save time and share extra memory pages.

          I think reading the first sentance and stopping covers my zygote, but adding the second sentance covers yours. So I think we're both right!

          I think both paths are useful. If your children need time to startup and become ready, spawn one that does start up work, and then it (pre)forks at the ready state to have processes ready to handle requests (your zygote). This does require a traditional fork() to avoid duplication of work.

          But if forking is expensive at runtime because you have a million FDs open and a whole lot of memory allocations, spawn spawners before you start doing work (my zygote). This could be unnecessary with a inexpensive way to spawn a new process from an process that has lots of resources in use.

          Of course, you can also use my zygotes to spawn your zygotes. Zygoteception.

          [1] https://chromium.googlesource.com/chromium/src/+/HEAD/docs/l...

          • skydhash 1 day ago

            I quite like the idea. I’m using OpenBSD on an oldish laptop, and fork-exec is expensive enough that it conflicts with the usb subsystem. Isochronous transfers have a 1ms realtime requirement and it seem that the fork-exec system calls hold the giant lock long enough to mess with it (audio stutters).

            While I’ve not bothered to profile it, but it seems that process that have lot of mapped pages is the issue (firefox, emacs,…). In the emacs case, the issue is when the main process trying to fork-exec, if I start a shell session (with shell-mode or term-mode), it works fine.

          • mpyne 21 hours ago

            > Oh I see. I guess your zygotes have developed more than mine. I think Google may have coined or at least popularized the term zygote for this in Chrome and Android, Chrome documentation [1] says:

            Google may have popularized the term, but this approach was already in use by KDE developers in the KDE 2.x timeframe, where it was used as part of a system called kdeinit.

            In this scheme, launching KDE apps from a KDE desktop could bypass much of the startup cost of dynamic linking by forking from a long-running kdeinit process (with kdeinit itself deliberately linked to all large dependency libs like Qt and kdelibs), dynamically loading the application logic (stored as a .so) and then launching the app.

            This was more to save startup time due to how long it took to dynamically resolve a multitude of C++-based symbols back then, all the common logic came before the app's own main() would ever be called. But it did also save a bit of memory as well.

      • PaulDavisThe1st 1 day ago

        > being able to inexpensively spawn a carefully tailored process regardless of the size and scope of the current process would be better.

        It's called clone(2)

        • trumpdong 23 hours ago

          Which argument to clone starts the process with an empty address space?

          • eggnet 3 hours ago

            That happens with execve(). clone() allows you to not copy the page table prior to the execve() call.

        • toast0 21 hours ago

          adding on the the sibling, what argument to clone allows me to set the fds of the child? AFAIK, you either share the FD table with the parent, or get a copy of it. If the parent has 1 million FDs open and the child doesn't want most of those, dealing with that has real costs. Many applications that tend to have large numbers of FDs and also fork/exec will mitigate the cost by spawning a process during startup that they can then use to spawn processes during runtime without doing it from the main process; this is a nice mitigation, but it shows a missing interface.

    • vlovich123 1 day ago

      The paper explicitly covers it that various memory COW/snapshot mechanisms are probably faster and safer than the zygote pattern. As it stands getting the zygote pattern correct and safe is something you have to plan for upfront. You can’t retrofit it which is why the paper mentions it has poor composability. Also the advantages of the zygote pattern can be overstated since the memory sharing benefit is minimal since it has to happen so early and modern OSes already transparently CoW duplicate pages in the background.

      • loeg 23 hours ago

        In what sense can you not retrofit the zygote pattern?

        • vlovich123 18 hours ago

          I recommend at least skimming the paper as it covers this. But essentially you can’t just inject a call at a random point in code to start being a zygote. It’s something you have to plan up front as to the exact point you’re going to fork and that you’re going to do it at the start of program before any threads have started or any files are open and before any locks have been acquired. It’s basically all the challenges of invoking fork at arbitrary points in time.

          The reason to do a zygote in the first place could be solved with alternative special APIs that are safer and harder to misuse. But we have fork so there’s not as big of a demand despite the warts.

    • p_l 23 hours ago

      And so easy to make into bottleneck.

      Yes, zygote pattern makes it easy to make fork() into bottleneck - it requires a lot more discipline and low level tricks (linker scripts, compiler-specific extensions, custom sections, low level dependencies on pagesize that get "fun" on ARM servers).

      If you don't, you might wake up with fork() causing latency issues.

    • cyberax 13 hours ago

      Unless you want to create a thread in your zygote. Then it breaks down.

      Raw fork() is terrible. Instead we need a proper primitive to stop and make a snapshot of a process.

      • pizlonator 1 hour ago

        You can create threads in the zygote. It doesn't "break down", but sure, there's a bit more work.

        My trick for that is that the set of threads that I create pre fork have to be suspendable and resumable, preferably lazily (they resume when they are actually needed). So, the zygotes are sitting with those threads suspended. When they become active, they can do work immediately. They might lazily resume those threads as needed.

        There are other idioms for this too.

        > Raw fork() is terrible. Instead we need a proper primitive to stop and make a snapshot of a process.

        Folks have been saying that it's terrible for as long as I can remember. But it's still there, because it's better than the alternatives

  • anarazel 1 day ago

    It is somewhat interesting that the most widely used "big" OS that doesn't use fork, i.e. Windows, has dog slow process creation...

    I agree that there should be non-fork primitives, I'm just not that sure that performance is the best argument.

    • pjmlp 1 day ago

      Because that OS best practices is to use threads.

      Traditionally Windows applications that create processes all the time come from UNIX heritage.

      Contrary to UNIX, Windows NT was designed with threads first mentality, from the get go.

      While on UNIX they were added after fact, and to this day there are gotchas mixing posix threads with signals, fork and exec.

      • zozbot234 1 day ago

        Windows was designed with threads-first mentality because on pre-386 machines you don't have viable process memory protection, so your tasks share memory by necessity. This is not a great argument.

        • epcoa 1 day ago

          This is not true. NT never had fork, was always based on the assumption of an MMU and Dave Cutler was a well known fork hater in the 80s long before this paper came out and made it cool to be so. By the time Windows 95 was out, the baseline was 386 with an MMU. CreateThread was initially designed for NT in 1993 though (which didn’t support pre-386 CPUs).

          • JdeBP 1 day ago

            As mentioned elsewhere on this page, Windows NT had fork from the start. Vide NtCreateProcess and what happens if an image file is not explicitly supplied.

            * https://computernewb.com/~lily/files/Documents/NTDesignWorkb...

            • dcrazy 1 day ago

              NtCreateProcess doesn’t accept an image file parameter.

              • JdeBP 23 hours ago

                You haven't read the doco. I did point to some. The image file is supplied (or not) via the section object.

                Think it through. Windows NT supported fork from the start in its POSIX subsystem, that subsystem was layered on top of the Native API, and this is the Native API mechanism that the POSIX subsystem employed. Although it took until Gary Nebbett for someone to publicly show how, even though people knew informally back in 1993.

          • keitmo 1 day ago

            NT performed unnatural acts to implement fork semantics for the POSIX subsystem.

        • pjmlp 1 day ago

          Windows NT!

          Misread on purpose to make a point?

        • dcrazy 23 hours ago

          NT was designed to be platform-agnostic, and its original target was the DEC Alpha. Its process model owes nothing to pre-386 CPUs. The WinAPI CreateProcess function is a layer atop NtCreateProcess, so that is where the pre-386 heritage lives. But even the WinAPI process model changed significantly with 32-bit Windows.

      • PaulDavisThe1st 1 day ago

        A more accurate way to describe this is that Windows' (NT onward) core execution context model is a bunch of threads that by default share memory, whereas Unixen have a core task context model of a bunch of threads that by default do not share memory.

        Both systems are implemented using threads as the execution context, but in Unix, the history means that that you fork+exec most of the time, resulting in a two tasks that do not share memory any more. By contrast, on Windows (NT onward) the common case when creating a new execution context is to create a thread that shares memory with others in its process.

        Both systems allow the easy use of the other's core abstraction. On Unix, you can either code like its 1986 and use fork without exec, or use clone(3) or any of its higher level abstractions like pthreads.

        You're right that POSIX semantics get tangled when using threads.

        • pjmlp 1 day ago

          Well, Windows before NT isn't the same design as Windows 16 bit, it only shares the name for all practical purposes, and has more influence from OS/2 than Windows 16 bit.

          Which is why I took the effort to explicitly refer to Windows NT on my comment, already expecting some traditional answers from UNIX folks.

          Also due to historical reasons POSIX threads are the outcome of every UNIX going their own way implementing threads, finally coming to an agreement years later, with all the plus and minus of relying in POSIX for portable code.

        • snozolli 1 day ago

          whereas Unixen have a core task context model of a bunch of threads that by default do not share memory.

          How are those not simply child processes? I don't understand your use of the word 'threads' here.

          Does the Unix world not distinguish between threads and processes? In Win32, threads exist within processes, and you can create new threads or child processes.

          • pjmlp 23 hours ago

            Actually on Windows a process is a thread with additional information.

            The unit of execution is the thread.

            On the UNIX world it depends on which UNIX you are talking about.

            Linux has a similar model to Windows NT nowadays, hence clone() as key primitive.

            Other UNIXes have different approaches.

            • PaulDavisThe1st 19 hours ago

              I worked on the kernel of DEC Ultrix, Mach/BSD and a couple of other early Unixen. The approach in all the ones I worked on was broadly the same.

          • trumpdong 23 hours ago

            They are child processes.

            Second answer: Linux doesn't differentiate between threads and processes. It has a "thread group ID" that serves a small number of purposes, and the rest of the difference is just whether the threads happen to share the same address space.

        • JdeBP 22 hours ago

          That's actually less accurate, not more. It's a post-hoc revision that conflates Unix with Linux.

          The Unix model was invented over a decade before the idea of multithreading percolated into mainstream operating systems at all.

          The reason that Windows NT started as it did, was that OS/2 had come out in 1987, with kernel threads, and the idea of multithreading had taken root. SunOS 5 gained threading, too.

          Windows NT applications development began with threading available as a mechanism from the start, and with a lot of people in the IBM/Microsoft world already knowing about its use in applications development from OS/2.

          Whereas with the Unices it came in more gradually, as the applications had often already been designed. The whole libthread versus libpthread thing made things interesting on SunOS for a few years, too. As did the first attempt (LinuxThreads) at providing threads on Linux.

          • thayne 20 hours ago

            PaulDavisThe1st is saying that the Unix pattern of forking a process (and not calling exec) was an early form of multi-threading (or multi-processing), but unlike threads in NT and later pthreads, they didn't share memory and communication between them required some form of IPC.

            • PaulDavisThe1st 19 hours ago

              Yep, absolutely corrrect. It was true at the lowest level (the semantics of fork) and it was true at the app/platform design level: in Windows you used threads inside a process, on Unix you used multiple communicating processes.

              This obviously changed as pthreads came into being, and at this point, I suspect that the typical use for threads-sharing-memory and threads-not-sharing-memory is the same on most platforms.

              A reminder that the task_t data structure describes threads and processes not just in Linux, but earlier Unixen also.

      • knome 1 day ago

        the only difference between a thread and a process on linux is how many structures they share. the function is identical.

        • pjmlp 22 hours ago

          Agreed, however not all UNIXes are like Linux.

      • sunshowers 1 day ago

        The problem is that threads are not fault boundaries but processes are. So they're not interchangeable when you care about resilience and misbehaving code.

        • pjmlp 23 hours ago

          True, but on Windows the approach is then to use COM servers, which have a faster IPC model, and can even serve multiple clients, depending on how the appartement space is configured.

          • mort96 23 hours ago

            "Faster IPC model" than what? Faster than writing to and reading from a pipe? Faster than POSIX shared memory?

            • pjmlp 22 hours ago

              Than UNIX fork/exec model, or calling into Create Process all the time.

              Windows has a more rich set of IPC stuff than POSIX, especially since it has a microkernel like design.

              If you are going to say it is everything on the same memory space anyway, it isn't.

              Optional on Windows 10, and enforced on Windows 11, Hyper-V is always running, and several components including kernel and driver modules are sandboxed into their little worlds.

              Several additional sandboxing changes were announced at BUILD.

              • mort96 22 hours ago

                fork/exec is not an IPC model...

                • pjmlp 20 hours ago

                  It actually kind of is, hence why you have information about parent/child and get to share memory.

                  This is how a http server back in the day would share the request context for the child process to reply back.

                  • mort96 20 hours ago

                    I would say that pipes and shared memory are the IPC mechanisms? Controlling the state of the exec'd process's file descriptors would counts as a way to set up interprocess communication, but once that's done, it's the pipe or SHM that does the actual communication.

                    • tliltocatl 3 hours ago

                      The problem with POSIX IPC is that passing file descriptors between processes (other than parent passing to child via fork) is hard. Yes, SCM_RIGHTS can do it, but it is quite error prone and rarely done.

          • dcrazy 20 hours ago

            If you want the isolation features of a separate process, you can’t substitute it with a single multithreaded COM server process.

            .NET tried this with app domains, which are now deprecated.

            • pjmlp 20 hours ago

              App Domains were in process, which isn't was I am talking about with outproc COM.

              Also App Domains are partially back in .NET Core, isolation features aren't there, but code unloading is, via AssemblyLoadContext.

              • dcrazy 19 hours ago

                My point is that “just write a COM server” is not an answer to the problem of “I want each work item to be segregated from each other.”

    • nvme0n1p1 1 day ago

      That's not the reason for the performance difference. Windows does have a fork primitive (ZwCreateProcess) and it's still slower than Linux's equivalent.

      • dcrazy 1 day ago

        Again, NtCreateProcess does not implement fork(). The fundamental characteristic of fork is that the child is an exact replica of the parent, down to the instruction pointer. Windows does not have a way to create a process object with such a configuration.

        Also, using the Zw prefix doesn’t make you look more knowledgeable, it makes you look like you’re trying way too hard to borrow credibility.

        • nvme0n1p1 21 hours ago

          Okay but people don't claim that copying the instruction pointer (a single machine register) is the reason for any speed difference. They claim it's due to the memory sharing. And that's easily disproven since you can share pages, just like on Linux, simply by passing null for the section handle, yet there's still a performance difference.

          Why does it matter which prefix I used? They both point to the same routine so my point applies either way.

        • netbsdusers 6 hours ago

          It's a completely uncontroversial fact that NT does implement fork(). Turn to page 183 of Helen Custer's "Inside Windows NT" and you will read about it.

    • aseipp 1 day ago

      I suspect it's a long tail sort of thing; it mostly doesn't matter except when it really matters. It's interesting that the stated motivation for the patch is in the context of agentic tools spawning subcommands. There's some related prior art in this area where the payoffs could be much greater, like fuzzing: https://gts3.org/assets/papers/2017/xu:os-fuzz.pdf is an example. It would be very interesting to see this patch applied to e.g. AFL++

    • mort96 1 day ago

      The problem with fork isn't really that it's slow. The problem is that if you want it to be not-slow, it locks you into a bunch of OS design decisions: you more or less need a memory subsystem where all writable pages are refcounted and copy-on-write when the refcount is bigger than 1, and you need overcommit.

      Now these decisions aren't objectively bad, but they have significant trade-offs and it's probably not a good idea that they're forced simply because we use fork()+exec() for process creation.

      • theK 1 day ago

        Didn't he just say that fork turns out to be comparatively faster to the non-fork samples we get? Ie Linux spawns processes faster than Microsoft's kernels?

        • nvme0n1p1 1 day ago

          We don't have any broadly used non-fork samples. Windows, macOS, and Linux all have fork. So the presence of fork can't be the reason for the performance difference.

          (Windows's fork is called ZwCreateProcess)

          • dcrazy 1 day ago

            NtCreateProcess does not implement a forking model. It is analogous to posix_spawn.

            • nvme0n1p1 21 hours ago

              If you pass null for the section handle, it shares pages with the calling process, thus implementing a forking model. Or at least the parts of a forking model that some people erroneously believe are responsible for performance differences.

          • Someone 23 hours ago

            MacOS has posix_spawn. See https://developer.apple.com/library/archive/documentation/Sy... (yes, that’s an iOS man page. MacOS has the call, too, but I couldn’t find the man page online and it looks identical to me)

            I don’t know how they implemented it, though. Under the hood, it could do the equivalent of a fork/exec pair.

        • mort96 1 day ago

          Didn't I just say that "the problem with fork isn't really that it's slow"? It's all the other OS design choices it forces on you if you want it to be fast.

          • theK 23 hours ago

            Right, you did. I somehow misread your comment.

      • marcosdumay 1 day ago

        CoW is probably a good idea whether you use fork or not. Or rather, fork is probably a better option than just exec exactly because it can benefit from CoW.

        At least on systems with virtual addressing. If you want to go into physical addressing, then yes, maybe it's a problem. But Linux will never touch anything with physical addressing, so I don't see what people are complaining about.

        • mort96 23 hours ago

          CoW is probably a good idea regardless, yeah. Overcommit is more questionable. Regardless, both ought to be argued based on their own merits. It's unfortunate that both are necessary as a consequence of fork().

          • mpyne 22 hours ago

            I don't think fork() mandates overcommit. OpenBSD doesn't seem to even allow overcommit or have an OOM killer, memory allocations that exceed available capacity fail immediately even if the memory is not touched.

            • vbezhenar 21 hours ago

              Let's say you have 1GB RAM. You're running program that occupies 600 MB. Now this program wants to launch second small program that occupies 1 MB.

              You're doing fork + exec.

              If you're overcommiting, fork will not reserve another 600 MB, and exec immediately after fork will cause total system usage to be 601 MB.

              If you're not overcommiting, that fork will fail, because total memory consumption will be 1200 MB which is more than 1GB. That somewhat restricts program design.

              • agwa 17 hours ago

                I think that on Unixes without overcommit, people allocate massive amounts of swap so that fork never fails.

              • cylemons 16 hours ago

                > If you're not overcommiting, that fork will fail, because total memory consumption will be 1200 MB which is more than 1GB. That somewhat restricts program design.

                Does this accounting apply to vfork as well?

                • cryptonector 13 hours ago

                  Correct: it does NOT apply to vfork().

              • dwattttt 12 hours ago

                > Let's say you have 1GB RAM. You're running program that occupies 600 MB. Now this program wants to launch second small program that occupies 1 MB.

                > You're doing fork + exec.

                This is the clear problem: you don't want another process that's a duplicate of the current one, that's just a detail of what you actually want: a 1mb process. Right now it's a badly leaky detail which you're forced to work around.

      • dapperdrake 1 day ago

        How else does consistency work, then?

        Only being half facetious here. Maybe you or someone else really has a better take.

        • mort96 1 day ago

          What do you mean by consistency here?

      • Someone 23 hours ago

        > The problem with fork isn't really that it's slow. The problem is that if you want it to be not-slow, it locks you into a bunch of OS design decisions: you more or less need a memory subsystem where all writable pages are refcounted and copy-on-write when the refcount is bigger than 1

        It may not be slow, but for the common case where fork is almost immediately followed by exec in the process where fork returns zero fork increases those refcounts and exec almost immediately decreases them again hand does typically unnecessary checks whether refcounts became zero). A combined fork/exec syscall can avoid that work.

        On the other hand, a sufficiently powerful combined fork/exec call has to have a lot of parameters that it has to check (whether to inherit open pipes, open files, setting the working directory, etc), and that slows it down.

        That can be avoided by having multiple variants of combined fork/exec calls, but you would need lots of them to cover all combinations of flags.

        I expect either approach should be faster then having fork, then exec as separate calls, especially when the process calling fork has many resources allocated.

        • thayne 21 hours ago

          Another possible design is instead of forking the current process, you create a new empty process, then the parent calls syscalls to set up the new process, and eventually call exec on the child process. That does mean you either need new syscalls for that, or adapt existing syscalls to take a pidfd as an argument. That also solves some other problems with fork/exec where the default is to inherit a lot of things you probably don't want. With this, you can opt in to inheritance instead of having to opt out.

          Or you could create a hybrid between a thread and a process, where it still uses the parent's memory space (unlike fok), but has it's own stack (unlike vfork), and is in its own process (unlike a thread). I think this is technically possible on linux, but there isn't a readily available interface for it. Although it seems like posix_spawn could be implemented that way...

          • dcrazy 20 hours ago

            Syscalls aren’t all that cheap either.

            • infogulch 14 hours ago

              io_uring taught us that if syscalls are expensive, queue them up in a buffer with one syscall to transfer the thread to the os to process it. So, queue up the new process mutations in a buffer with a single syscall to process all of them in a batch. This model should have replaced repetitive syscalls across the kernel years ago.

            • thayne 14 hours ago

              This true, but these methods don't increase the number of syscalls you need to make.

          • fc417fc802 14 hours ago

            > you create a new empty process, then the parent calls syscalls to set up the new process ...

            That does seem like a much better design to me. But I wonder if that was considered way back at the dawn of computing and rejected for good reason?

            > I think this is technically possible on linux, but there isn't a readily available interface for it.

            Yes there is, see `man clone`. POSIX and glibc are quite different from the kernel in this regard. AFAIK under linux there are just threads of execution that might or might not share various namespaces and memory mappings. That said, the kernel does place a few artificial restrictions on what combinations are allowed in order to (as I understand it) guard against the unintended exercise of entirely untested combinations that serve no known practical purpose.

            The practical problem is that if you start doing as you please with the various namespaces and mappings you quickly become incompatible with glibc and by extension most likely the majority of the dynamic libraries available on your system.

      • foresto 23 hours ago

        > The problem with fork isn't really that it's slow.

        Did someone suggest that it was?

        • mort96 23 hours ago

          anarazel's comment focuses entirely on performance, indicating that they have an impression that the discussion about why fork is bad is about performance. I'm not entirely sure where this impression came from, as it's not mentioned in rom1v's quote nor a point in the linked paper, "A fork() in the road".

      • adgjlsfhk1 22 hours ago

        One os level thing that is interesting to me is if it would be possible/wise to make an OS based on (concurrent) garbage collection.

      • thayne 21 hours ago

        With large enough processes, like say a server JVM process that uses 10s of GBs of RAM, even just copying the page tables for CoW can be slow. And unless you have aggressive overcommit settings you can get an OOM on fork, even if you're just going to exec something small.

        vfork helps a little, but it has a lot of restrictions on what you can do before the exec, and on unix that's basically the only place you can do things like close files, change signal masks, drop privileges or set up seccomp, etc.

        • cryptonector 13 hours ago

          vfork() helps a LOT. The restrictions on what you can do on the child-side of vfork() are pretty much the same ones as for fork() + you must not do anything to damage the stack frame of the vfork() caller (i.e., you can't return).

      • tliltocatl 20 hours ago

        In addition to what you said: forking from a process running on multiple cores is slow once you have mark all pages as read-only and shoot this out to all cores. TLB synchronization is super expensive. Unix originally didn't support threads (want concurrency? just fork!) but with modern multicore that's clearly unsustainable.

      • emmelaich 13 hours ago

        The nice thing about fork+exec is that's its simple and flexible.

        To avoid the problems, see roc's comment under the article. Esp use of a zygote process.

      • netbsdusers 6 hours ago

        Solaris and Windows NT both have fork() and strict accounting by default.

  • Animats 1 day ago

    > The received wisdom suggests that Unix’s unusual combination of fork() and exec() for process creation was an inspired design.

    No, it was done that way so that you could launch a program that was too big to fit in memory with the parent program. The original implementation worked by swapping out the forking program to disk on a fork() call. Then, at the moment the program was swapped out but control had not returned, the process table entry was duplicated and adjusted so that there were now two processes, one in memory and one swapped out. The one in memory then got control, and could do an exec() call.

    This allowed large programs to run on small PDP-11 machines. It was needed back in the era of really expensive memory. That's why.

    QNX had an interesting approach. Program loading isn't in the OS at all. There's "fork", but program loading is in a library. It links to a .so file which reads the executable header, allocates memory, loads the program, gets it ready to run, and starts it. The program loader runs in user space and is unprivileged. This is probably the right way to do it.

    • lukan 1 day ago

      It is almost as if you agree with the authors ..

      "In this paper, we argue that fork was a clever hack for machines and programs of the 1970s that has long outlived its usefulness and is now a liability"

      (But thanks for the good explanation)

    • dcrazy 23 hours ago

      Don’t pretty much all OSes implement process startup in userspace? On macOS, the kernel creates a process with an image of dyld and points it at dyld_start, which actually takes care of parsing the Mach-O header. I assumed ld.so does the same job on Linux.

      • purkka 21 hours ago

        Nope, the kernel can load static ELF binaries. ld.so is only needed for dynamically linked binaries, and in fact many Go applications (for example, as they're statically linked) ship as containers with nothing but the single binary.

        • dcrazy 20 hours ago

          Thanks. I completely forgot about static binaries.

          • loeg 20 hours ago

            Of course ld-linux itself is an ELF binary. The kernel loads it.

            • bregma 6 hours ago

              Not only is it an ELF binary, but it is ironically a static ELF binary.

        • krackers 12 hours ago

          You can do this on macOS too, if you're willing to break all forward/backward compatibility and make direct syscalls you can have a purely static binary. Without the LC_LOAD_DYLINKER command on the mach-o binary the kernel should just jump to the entrypoint based on LC_UNIXTHREAD. (This may not longer work on arm machines though if they actually trap on direct syscalls not through libSystem, similar to the BSDs)

    • not_a_bijection 22 hours ago

      I think fork() is more of a PDP-7 mistake than a PDP-11 mistake. On the original UNIX system, memory was so limited that the only sane partitioning was to write the running program's memory image to disk, then reuse the running image as the child. An immediate consequence is the UNIX I/O model, where disk I/O is always synchronous (can't swap processes while waiting for disk I/O because swapping processes requires disk I/O). Anyway, as soon as the UNIX group got a PDP-11, the model broke down, because they had enough memory for multiple processes, but fork() didn't allow them to run concurrently, because their first PDP-11 didn't have an MMU. So they whined until they got one with an MMU instead of fixing their broken design.

    • duped 22 hours ago

      > It links to a .so file which reads the executable header, allocates memory, loads the program, gets it ready to run, and starts it. The program loader runs in user space and is unprivileged. This is probably the right way to do it.

      aiui this is what exec does, the problem outlined here is the split between process creation (expensive, kernel space, has to be done each time even if spawning the same process "template" repeatedly) and loading (cheap and in userspace).

    • bluepuma77 22 hours ago

      > It was needed back in the era of really expensive memory.

      Well, it seems we are back in an era with really expensive memory.

    • afiori 18 hours ago

      This comment starts with a no, but agrees with the parent...

    • BobbyTables2 15 hours ago

      The QNX approach is also pretty much how the dynamic linker loads shared libraries today in Linux .

      “An era of really expensive memory”. That sounds familiar…

      • vanviegen 10 hours ago

        I think GP was saying that in QNX the spawning process was responsible for dynamically linking it's child process before running it. With Linux, I think it's the spawned process taking care of it's own dynamic linking.

        • bregma 6 hours ago

          On QNX the process spawning is done by sending a message to the userspace process manager, which creates a new process table entry and queues up its initial thread. When its initial thread gets a timeslice its entry point may be the dynamic loader (as specified in the PT_INTERP segment) which then does all the dynamic linking as the spawned process or it might be some other entry point like with a statically-linked executable.

          So on QNX, the spawned process does all the dynamic linking. The spawning process just sends an asynchronous message to the process manager and then gets on with things in a very deterministic manner as befitting a hard realtime system.

    • cryptonector 12 hours ago

      Cygwin's fork() is similar to what you describe for QNX.

      • JdeBP 10 hours ago

        It's a fairly widespread idea for architectures that try to move things out of kernel mode. The Hurd does program image file loading in userspace, too, in its exec server(s).

        The tricky part is setting up the initial process. The way out for that is static linking and re-use of the fact that the operating system kernel loader has to understand and be able to load (at least a small subset of) program image file formats too.

    • cryptonector 12 hours ago

      > > The received wisdom suggests that Unix’s unusual combination of fork() and exec() for process creation was an inspired design.

      > No, it was done that way so that you could launch a program that was too big to fit in memory with the parent program.

      Ironically vfork() is even better in this regard. I wish Unix had only ever had vfork().

    • derriz 8 hours ago

      But why is having a pair of separate independent operations, fork and exec, required to achieve this? A single fexec call could be implemented to work in the way you describe, no?

  • up2isomorphism 19 hours ago

    Not sure if fork is outdated or not, but people calling it a “hack” obviously have pretty bad engineering taste.

sanderjd 1 day ago

I just ran into this recently, where I had an obscure bug caused by needing to close more file descriptors in the forked process. "I want a clone of the current process" is just way less common in my experience than "I want a completely new process". It feels crazy that we don't have a way to directly express the latter thing, and can only approximate it by cloning and then fixing things up in post.

  • dnw 1 day ago

    What do you mean by "a completely new process"?

    • sanderjd 1 day ago

      A process that shares nothing with the process that spawned it.

      • jerf 1 day ago

        A thing that makes that complicated is that while you want that conceptually, you don't want that in reality. For instance, if the spawning process is in a container of some sort and it spawned a process that "shares nothing with the process that spawned it", the spawned process would no longer be in that container, because the state of "being in the container" is one of the things it shares with the parent process.

        This is just an example of I don't even know how many things a modern-day process will share from its parent.

        By "complicated" I do not even remotely mean "unsolvable". I just mean that if you really dig down into what it means to "share nothing" in a modern operating system, it's a lot richer than it was back when fork+exec was a practical solution. There's a lot of fuzzy things that could go either way when you say "shares nothing".

        • dcrazy 23 hours ago

          It’s such a bad idea that every OS except Linux implements it? On macOS it’s posix_spawn, on Windows it’s NtCreateProcess.

          • jerf 23 hours ago

            Who said anything about it being a "bad idea"?

            I also explicitly said this wasn't unsolvable. My point isn't about technical implementations or code, my point is that the casual "I want to share nothing about the parent process" thought in sanderj's mind, and presumably a lot others, is much more ill-defined than they realize. There's a lot more state that a process has than what file descriptors are open in a modern system.

            Moreover, as things like "in which container is this running" demonstrate, those are also not "create a process that has nothing to do with this process", because, again, there's a lot more to "having to do with this process" than "what file descriptors are open".

            Also, as the name might have been a clue, Linux has posix_spawn: https://linux.die.net/man/3/posix_spawn. It also has a thing called "clone": https://www.man7.org/linux/man-pages/man2/clone.2.html Nor do I claim this paragraph is an entire overview of all the ways of starting a process in Linux. If you want to understand what I mean by "lots of details in a modern OS", your assignment is to carefully read the entire "clone" man page, and you'll start to see what I mean, though I'm not sure even that is all the state associated with a process nowadays.

            • dcrazy 23 hours ago

              Linux posix_spawn is a wrapper around clone and exec. There is no primitive on Linux to create an entirely blank process. This is adequately discussed in the linked LWN post.

              Other operating systems either have parallel APIs to fork (e.g. the posix_spawn syscall on macOS) or do not provide fork at all (Windows).

              • jerf 4 hours ago

                You seem to persist in reading into my words claims that aren't there and then excitedly debunking them. I feel I'm extraneous to this process, though, so I think I'll let you carry on arguing with the guy in your head on your own terms. It's more fun for both of us.

            • sanderjd 21 hours ago

              It's not a casual thought. I recognize that there are lots of details, there always are, we're talking about computers :)

              I don't think it is necessary (or the best implementation) to clone the parent process, in order to maintain important properties like the process tree / container state, etc. I recognize that it's a sorta neat hack, "well if we just start by cloning the parent, then we don't have to figure out what state to include!", but that just pushes the details to the child process needing to figure out what to exclude, which IMO is a worse default.

        • sanderjd 21 hours ago

          Yes, stipulated. And it it's true that we should have a primitive for spawning a completely new process, because that's what we usually want. I agree that the details are both non trivial and soluble.

      • JoBrad 1 day ago

        That’s how you get zombie processes and memory leaks.

  • 1718627440 1 day ago

    But you generally want to communicate with that process, so you do need to setup e.g. file descriptors and stuff, which needs information from the parent process to be passed.

    • jonhohle 1 day ago

      Most programming languages abstract this out to be able to connect or drop the 3 standard pipes. Typically this is the only thing that can be shared anyway unless the other program is specifically shared and expects other file handles to be available, in which case fork might be the right system call anyway.

      • sanderjd 21 hours ago

        Right. It's not that fork is useless, it's that it's weird that it's the only way to do this thing that it isn't particularly well suited for.

    • stefan_ 1 day ago

      Keep in mind that this is the only way to start any process. Even if you just want to launch some throwaway utility program.

    • yxhuvud 1 day ago

      Yes, you do want to pass in some stuff. But by default you get every single open file descriptor and a copy of every single stack that any threads use for execution.

      It shares way too much, and have huge use cases where it is really, really bad.

    • sanderjd 21 hours ago

      Nevertheless, inclusion would be a better default than exclusion in most use cases I've ever had for process spawning.

    • gmueckl 15 hours ago

      A variant of exec could take an initial table of file descriptors in the current process that get cloned into the new child. Pipe creation could also get rolled into this mechanism. That should take care of the most obvious leaky bit of fork()/exec(), at least.

  • stabbles 1 day ago

    Isn't that covered by O_CLOEXEC?

    • anarazel 1 day ago

      There's a bunch of nastiness around that too. If you have e.g. library state that assumes the fd still works you can get her very confusing bugs once another file is opened into that fd number...

      • JdeBP 1 day ago

        You may be mixing up fork and exec. Library data state isn't retained over execve(), and O_CLOEXEC does not take effect at fork().

        • anarazel 1 day ago

          Indeed. Not enough coffee, apparently.

    • sanderjd 21 hours ago

      I think it is error prone to need to iterate file descriptors and set this in order to inherit nothing. Excluding by default would make sense IMO.

  • 7jjjjjjj 1 day ago

    >It feels crazy that we don't have a way to directly express the latter thing

    Isn't that what posix_spawn is for?

    • yxhuvud 1 day ago

      And how do you think posix_spawn is implemented?

      • JdeBP 1 day ago

        This is an oft-overlooked point. An obvious place to look for improving fork+execve is to see whether posix_spawn can be given more efficient kernel mechanisms to be based upon.

        And of course that has already been done. On NetBSD, posix_spawn() is a fully-fledged system call and much of the work is done in kernel mode.

        * https://blog.netbsd.org/tnf/entry/posix_spawn_syscall_added

        • dcrazy 23 hours ago

          This is literally discussed in the article this post links to.

          • JdeBP 23 hours ago

            Not really. They didn't get anywhere near as far as noticing the prior art of NetBSD, not even on the mailing list discussion behind that article.

    • toast0 1 day ago

      posix_spawn addresses the need from userspace. Under the hood, it's still doing more or less a fork/exec, with the baggage that comes with it. A syscall would be nicer.

mrkeen 1 day ago

> fork() is a relatively expensive system call; it must copy the entire process state (including memory) for the child process. Many optimizations have been made over the years, but a fork is still a fundamentally costly operation. To make things worse, a fork() call is often immediately followed by an exec(), which will discard all of that memory that was so carefully copied for the child.

It's weird to leave out a mention of copy-on-write - the optimisation that means that you don't copy over all the memory.

  • FooBarWidget 1 day ago

    It says state. Copy on write still means it's O(number of page table entries) even if you don't copy the contents. It's a well known issue that forking a program with large virtual memory size is slow.

    • mort96 1 day ago

      It says "(including memory)". It's pretty natural to read this as "(including the contents of allocated pages)".

    • m00x 1 day ago

      On modern hardware a cow page copy should only take 1-5ms. Redis forks to save the db to disk and it's been a solid design choice.

      I guess it depends on how sensitive your application is to main thread pauses.

      • trumpdong 23 hours ago

        So like 1000-5000s if you have 4GB of data? Over an hour?

      • Joker_vD 18 hours ago

        Redis absolutely suffers from long-executing fork() in practice, its developers even griped about it a couple of times on their blog.

      • tempest_ 17 hours ago

        I have found that design choice to be annoying

  • tux3 1 day ago

    This was left implicit in the article, but what they mean by copying the process state here is the memory management structures. That's mainly the page tables and the VMAs.

    That means you have to allocate new pages to hold a copy of all these structures, even if the actual memory pointed by the pages is shared. And walking all those structures to make a copy is still costly.

  • cls59 1 day ago

    Even with copy-on-write, fork() still has to pay the setup cost for COW. If the parent process has a lot of busy threads (e.g. Java), you can end up doing a lot of unnecessary COW before exec() fires.

    • josefx 22 hours ago

      Isn't that what vfork tried to address? No COW, the child starts in its parents address space and only gets its own after calling exec.

      • j16sdiz 15 hours ago

        Yes, the next sentence in TFA is:

        > Attempts (such as vfork()) have been made over the years to optimize for this case, but the pattern still is more expensive than it could be.

        Basically vfork do a "stop the world".

        • cryptonector 13 hours ago

          > Basically vfork do a "stop the world".

          vfork() does NOT stop the world in many / most implementations. The ones that do stop the world do it because someone misunderstood the whole "vfork() stops the parent process" -- yes, it stops the parent process in a pre-threads world, but it doesn't have to stop any other threads but the one that called vfork(). Indeed, many implementations don't do that.

          (Someone once tried to make NetBSD's vfork() stop the world because that's what the pre-threading man page said it does. I did my utter best to keep that from happening at the time, and it didn't then. Hopefully no one tried again later.)

  • epcoa 1 day ago

    > It's weird to leave out a mention of copy-on-write

    For the intended audience of such a paper this is base knowledge.

  • thamer 22 hours ago

    Redis is the kind of process where this matters a lot, and while fork() doesn't copy the memory, it still needs to copy the page table. For a process holding tens of GBs of RAM, fork() can take a long time, and there's one every time Redis dumps its .rdb file or rewrites its binary log ("AOF").

    Even back in 2012 this blog post showed the high cost of this operation: https://redis.io/blog/testing-fork-time-on-awsxen-infrastruc...

    On an m2.xlarge using ~25GB of RAM, fork() took 5.67 seconds. That's a long pause when Redis clients typically experience single-digit msec latency for most operations. Yes, that's only the time needed to copy the page table. It's surprising they don't mention huge pages, it seems like it would be a key consideration here.

    No doubt hardware is faster 14 years later, but Redis instances likely use more RAM too. It'd be interesting to see this benchmark revisited.

uecker 1 day ago

The elegance of the fork() + exec() model is that every kind of configuration can be done after the fork using all the usual APIs. Every attempt to replace it with a combined call that I have seen so far seemed fundamentally poorer because it needs to add all configuration options as parameters to the call and then do this in away that you can extend it later and does not become a mess.

  • amluto 1 day ago

    I have the entirely opposite opinion. IMO a big mistake of the UNIXy model is that so much state is preserved across the creation of a process. For example, there are APIs to have a specific thing be fd number 4 so you can run a program and have it find that thing at fd 4. This is weird.

    Windows, for all its many, many faults, did not use fork+exec and instead mostly has options for how one creates a process. It wasn’t done elegantly, but it was the right decision.

    • 1718627440 1 day ago

      Is it weirder, that you can pass an variable precisely into argument 4? You do need to pass information to a subprocess and there needs to be some agreement on what means what. Sure, maybe you could use names instead of fds, but that sounds needlessly complicated.

      • jonhohle 1 day ago

        That’s like saying you could use positions to specify function argument access (as in assembly) instead of variable names. File descriptors being numbers that are likely array indexes in a file handle seems like a leaky abstraction. Having a namespace that a parent process share with its children seems like a much cleaner design.

      • amluto 1 day ago

        A way to pass a defined list of handles to a subprocess (or a friendly other process) makes sense. Having that mechanism be direct inheritance of those handles with the same numbering as the source is obnoxious.

    • burnt-resistor 1 day ago

      You're simply failing to grasp the value of the simplicity, compatibility, and portability of POSIX/*nix. Inventing yet another way to create a process would be complex and break things. It's a-la-carte to enable configuration after fork of the new CoW or non-CoW process but before exec (unless vfork or similar were used). This is the model.

      If you want to greenfield re-engineer the world with all new system calls and a totally different execution model, feel free to go right ahead.

      • wvenable 1 day ago

        "The reasonable man adapts himself to POSIX: the unreasonable one persists in trying to adapt the POSIX to himself. Therefore all progress depends on the unreasonable man."

        ― George Bernard Shaw, probably.

    • __david__ 1 day ago

      Having fd 4 mean something specific is no weirder than having fds 0,1, and 2 mean something specific, which is probably never going to change. At some point you just gotta embrace the Unix.

    • chasil 1 day ago

      Well, Cygwin and Busybox have shown me that fork-heavy activities are about 100x slower on Windows than Linux.

      The Windows approach may be correct, but it suffers in performance from the POSIX perspective.

      I have heard that WSL1 iimproves this.

      • amluto 1 day ago

        Linux has worked pretty hard to optimize fork(). This doesn’t mean that fork() is a good idea.

        Windows does not historically depend on fork(), so there was no native fork(), so Cygwin kludged it up.

        • JdeBP 1 day ago

          Actually, there is a native fork. There had to be, as POSIX personality support was a part of the Windows NT 3.1 design. What there wasn't was a Win32 form of fork. The Native API for Windows NT allowed it quite straightforwardly.

          • jkrejcha 19 hours ago

            Iirc Cygwin used to use it but iirc they moved away from it because they said that it was pretty slow

            Though actually iirc werfault uses NtCreateUserProcess() to clone processes when writing out crash dumps to this day

    • uecker 1 day ago

      Well, a lot of the power of the UNIX shell comes form this and I see this as a major advantage over Windows. So no, I do not think Windows got it right.

      Any kind of replacement should aim for the same conceptual simplicity and power. Sadly, I fear that people driving development nowadays are more interested in building unbreakable walled gardens for advertisement or app stores, or trying to squeeze down the some small gain when used on the cloud. I am more interested in general computing on the user side.

      • dcrazy 20 hours ago

        Nothing about the UNIX shell is reliant on the fork model. Windows processes have stdio handles as well.

        • uecker 20 hours ago

          A lot of features of UNIX shells are build around pipe and dup and the fork + exec model. One can certainly implement in differently, but it is - like UNIX in general - very nice and elegant.

          • amluto 18 hours ago

            Help me out here, please. Off the top of my head, the exec command is dependent on exec, except that a spawn + wait implementation would be a mostly okay substitute.

            Pipes and redirections don’t need fork + exec. Neither do subshells.

            • uecker 12 hours ago

              If you use pipe() you get two ends in the same process, then you fork and child and parent can communicate. This is how a unix shell setups up pipes and it is rather elegant.

              • dwattttt 5 hours ago

                Doing the same thing on Windows, I create the pipe and get two ends in the same process. Then I'd call CreateProcess and indicate I want the pipe's handle (fd) inherited to the child, and I'd use a prearranged way to tell the child what the fd value is it should use.

                Possibly the most common way to tell the child the value is by setting it as a CLI arg in CreateProcess.

                • uecker 4 hours ago

                  Yes, and CreateProcess needs special facilities to make this possible while in the UNIX model you don't.

          • IshKebab 10 hours ago

            It's an elegant hack, but it's still a hack. Not what we should be doing in 2026.

            • uecker 3 hours ago

              I would prefer to continue to use elegant interfaces even beyond 2026.

              • IshKebab 3 hours ago

                I would prefer to use elegant interfaces that aren't massive hacks.

                • uecker 3 hours ago

                  Define "hack".

                  • IshKebab 2 hours ago

                    Something that works but is a surprising and suboptimal way to do things.

                    I dunno, that's the best I can do for now. Maybe you can do better?

                    • uecker 1 hour ago

                      I don't think it is hack. I think it is a nice and clean API and the hate is largely irrational. I think one could improve usability for multi-threaded programs though.

  • fanf2 1 day ago

    Yeah. The right way to eliminate fork() is to make the usual APIs that modify process state take an explicit process handle, so the same APIs can be used to set up an empty process. They can also be composed in other ways, eg for IPC or debugging.

  • garaetjjte 1 day ago

    That's mostly papering over design mistake that most syscalls doesn't accept target pid. Otherwise you could just create suspended process, configure it with syscalls that explicitly take target pid, and start it.

    • uecker 1 day ago

      Maybe, I am not saying fork() + exec() model couldn't be improved, but most people saying it is "terrible" and it needs to die seem to go on to propose something substantially worse.

    • trumpdong 23 hours ago

      Or have a syscall that runs any other syscall in a different process.

  • __david__ 1 day ago

    I agree. I think the current way is very nice to use (in c). I think the best way would be to have something similar to vfork() but not bound by posix rules. Then make the normal posix apis (close, setuid, etc.) act like the Rust “builder” pattern. Possibly giving them a prefix for explicitness. That way the “fill out a giant structure” people could have their wish and the people that just want a faster posix experience don’t have to learn an entirely new concept and api surface. It would be future extensible that way, too (just add more prefixed calls to the builder).

  • PaulDavisThe1st 1 day ago

    Whatever elegance fork(2) has (or doesn't) have, clone(2) has more.

  • matheusmoreira 1 day ago

    The new system calls described in the article have an extensible declarative command interface built into them to do things like close or duplicate file descriptors. Not opposed to it but it definitely stood out to me.

  • jcranmer 23 hours ago

    Calling that elegant is a path dependence of the history of fork+exec.

    In an alternative world where fork+exec never existed, a lot of those "usual APIs" would probably have had an explicit pid argument to them that let you modify process configuration from a different process. (This is how Fuschia works, e.g.). There's a lot of benefit to this world: the most obvious is that you don't have to magic up some IPC system just to report configuration errors, but there's actually a good amount of utility in being able to have a manager process that is tweaking attributes of its children (e.g., debuggers would love it).

    • uecker 23 hours ago

      Weren't there enough parallel paths of development in this world?

    • trumpdong 23 hours ago

      Or you could call ptrace_syscall (that doesn't currently exist) on your child processes that you are tracing because you'd always be tracing them by default, or get an io_uring for the child process, or...

      • uecker 23 hours ago

        A ptrace_syscall would be interesting and would seem to be a full replacement for having the pid argument everywhere.

        But frankly, I am not really seeing the value.

        • trumpdong 20 hours ago

          The value is not needing to change every other syscall and not needing to write new ones with a pid argument (besides which, what when you want to change it to a pidfd argument? then you add pidfd_syscall instead of duplicating every syscall again)

          • uecker 7 hours ago

            I meant the value of running syscalls in another process from the parent process in contrast to (v)forking and running them in the child directly.

            • trumpdong 3 hours ago

              The value is starting the child with a clean slate instead of a copy of its parent.

  • trumpdong 23 hours ago

    It should be spawn, configure, exec. Configure can be done if the process starts with a ptrace attachment and no threads, so you can force it to do syscalls. Linux doesn't even have a concept of "process with no threads", so it'd probably have to have a dummy thread.

  • pjc50 23 hours ago

    The flip side of this is that you have to be aware of the entire state of the process, including everything done in libraries, in order to correctly start a new process.

    Quick, what's the highest numbered open file descriptor in the your program?

    This gets even worse if you have multiple threads running. Without looking it up, what is the state of all the various synchronization primitives in a forked process?

  • jkrejcha 20 hours ago

    I kinda disagree, though I do see the usefulness here. While fork/exec can be useful in some cases, it'd be honestly pretty neat if the APIs took a pidfd argument (maybe with 0 meaning current process). Only program is setuid/setgid binaries I suppose but maybe this case is better handled by special casing `exec`.

    For example

       pidfd_t ps = spawn(); // creates a process stopped (kernel does this anyway by default)
       setuid(ps, 33);
       capset(ps, ...);
       socket(ps, ...);
       mmap(ps, ...);
       process_vm_writev(ps, ...);
       exec(ps, ...);
       signal(ps, SIGCONT);
       // error handling elided
    

    I guess this is a little bit me being a bit of critical of the usual syscall APIs for not thinking about "what if I want to do this to another process I have access to" but...

    It also makes things like thread safety even reasonably doable with fork. I do agree though that stuff like CreateProcess which take in a gazillion parameters don't really make for the greatest of userspace APIs

    • uecker 20 hours ago

      Maybe, a few people proposed this. It is a lot better than a single spawn call.

      But how often would one actually need this? And what are the semantics? Refer arguments (e.g. file descriptors) to the current process or the other one? How are cross-permissions handled? It seems a lot of complexity...

      Someones proposed a ptrace_syscall which could achieve the same thing.

      • jkrejcha 19 hours ago

        > But how often would one actually need this?

        Well, the idea is that it'd probably be close to the default API for spawning processes (and could even be the bedrock for posix_spawn and friends in libc (and potentially even "simple" fork cases[1])). fork/clone would be the special case

        In most cases, most programs don't need special setup. Something like `ptrace_syscall` would also work for this and would be probably the way to do it with the backwards compat limitations of nowadays

        ptrace-ability seems to be generally how permissions for this sort of thing are handled in general (see also procfs, process_vm_writev, ptrace, etc). The complication is a little bit around setuid programs but either you could special case execve to imply SIGCONT for setuid or have execve also imply a SIGCONT as well

        [1]: Probably would be rare for a compiler to optimize it though

  • zzo38computer 17 hours ago

    I agree with it, although still the fork is expensive like they mention. There is clone with some flags, although that does not really solve it.

    I think one problem is that it is already how it is; making an entirely new operating system (that is not Linux, not GNU, and not POSIX) would solve it, but that is not the case here, so it would need to be done as it is.

    One possibility would be a new function that creates a new empty child process, but the parent process specifies what system calls the child process executes, and can stop if specifying that exec or exit is (successfully) called by the child process, or if the parent process gives it the program memory to execute directly instead of using a file (since that use is also useful). The new function can still have some of the clone flags available. (I don't actually know how much better it would work.)

    There are other possibilities as well.

    The existing methods can also remain available for when they are helpful, but functions such as popen might be changed to use the new method.

  • fonheponho 4 hours ago

    > The elegance of the fork() + exec() model is that every kind of configuration can be done after the fork using all the usual APIs.

    Unfortunately, the opposite is true, when the parent process is multi-threaded. In the child process, only one thread exists (the thread returning from fork()), but the memory is an exact copy of the parent's. As a result, the child may inherit locks (resident in memory) that are in acquired state, but have no owner threads -- the threads that are responsible for eventually releasing those locks in the child's copy of the process memory do not exist in the child. If the single thread in the child process (returning from fork()) attempts to take such a lock (before exec), it deadlocks. This is why POSIX says that only async-signal-safe functions may be called in a child process, between fork and exec. And then, for example, "malloc" is not such a function (at least per POSIX), so the fork-to-exec environment in the child process is an extremely uncomfortable one. You've got to preallocate everything in the parent, can't report errors to stderr, etc.

    https://pubs.opengroup.org/onlinepubs/9799919799/functions/f...

    https://pubs.opengroup.org/onlinepubs/9799919799/functions/V...

    The fork(2) Linux manual page spells out the sam restriction.

    https://man7.org/linux/man-pages/man2/fork.2.html

    https://man7.org/linux/man-pages/man7/signal-safety.7.html

    "pthread_atfork" exists, but is effectively unusable.

    https://pubs.opengroup.org/onlinepubs/9799919799/functions/p...

    • uecker 4 hours ago

      Yes, threads are a complication, but this still not "the opposite".

jcalvinowens 1 day ago

It is a weirdly common misconception that that fork() is cheap... it is O(N) on the size of the process, and it always has been.

Yes, it's copy on write... but there is a linear relationship between the size of the process and the number of page table entries required to represent it.

  • themafia 12 hours ago

    > the number of page table entries

    This is not exactly fixed since you can vary the amount of memory each page maps with things like hugepages and the same process can run with different page sizes.

    • IshKebab 10 hours ago

      You can in theory, but it's rare in practice because it isn't always enabled and it requires root to configure.

ajkjk 1 day ago

Fork always seemed conceptually terrible even when I first learned about it.. If you want to do one thing (start a process) you should not have to use a mysterious incantation that does a different unrelated thing (forks your process) in order to do it.

I am curious about what the best way to handle the example in the article of one process spawning many git subprocesses is. Surely it just doesn't make sense to repeatedly start git from scratch in the course of a long-running parent operation. What's the low cost abstraction for the same result, though?

  • wmf 1 day ago

    libgit2 exists. You could imagine communicating with some gitd over a pipe/socket but I don't know why that would be a good idea. Short of that you have to spawn processes.

    • trumpdong 23 hours ago

      On Windows maybe it would be a COM server, using IPC built into the OS. The client sees it like a local function call.

  • spacechild1 22 hours ago

    Yeah, as someone who originally came from Windows, the fork+exec model never made sense to me. Now I know it's just a historical quirk, but for some reason there are still people who pretend that fork+exec is actually a good thing...

  • kps 20 hours ago

    Fork is conceptually simple. Without bringing in any other layers, you start a process with the one thing known to exist: yourself.

    Otherwise you need multiple steps to create a process, fill it with something to run, and arrange for it to execute. Or like Win32 you permanently smush them together with other layers, like filesystems and object loaders and linkers.

    • Too 13 hours ago

      Fill with what stuff exactly?

      The only thing I want to inherit from the parent process is its cwd and environment variables, even those are often overridden. The rest can easily be passed explicitly through other channels like pipes or command line arguments.

      Back to the example from the article. It makes no sense that a git-subprocess forked from a web server need to have any process state inherited from the web server.

      • kps 2 hours ago

        > Fill with what stuff exactly?

        Yes, exactly. Cloning, as a process creation primitive, is the one thing that doesn't need to be concerned with other stuff.

        > … a git-subprocess forked from a web server …

        That's pulling in a whole load of assumptions that are distinct from process creation. You can have processes in an environment that has no concept of file system or persistent storage at all.

    • ajkjk 13 hours ago

      I gues that way of thinking makes sense if you have a certain model of what a process is, in terms of the data structures and runtime state etc. But, tbh, I think of processes as glorified function calls, which happen to have that stuff involved as an implementation detail. And if spawning a process call is supposed to act like a function call, then of course it should not inherit state. You should call the function you want to call, not call yourself with an instruction to switch over to it instead.

      • fluffybucktsnek 9 hours ago

        Conceptually, processes are more akin to units of isolation. Threads are closer to function calls.

    • IshKebab 9 hours ago

      It's not conceptually simple. No other object creation API works by copying an existing thing and then modifying it. You don't create a new file by copying an existing one and then modifying it. You don't create a new window by copying an existing one and modifying it.

      Attempting to justify clone/exec as a reasonable design is just Stockholm syndrome.

      • kps 2 hours ago

        > No other object creation API works by copying an existing thing and then modifying it.

        Clone-and-modify is pretty common in CAD.

        > You don't create a new file by copying an existing one and then modifying it.

        Clone-and-modify is almost universal in version control systems.

        • IshKebab 2 hours ago

          > Clone-and-modify is almost universal in version control systems.

          It's closer to copy-on-write. Also, it actually makes sense there because in 99.999% of cases a commit actually is a modified copy of its parent. That isn't true for process spawning.

mpweiher 9 hours ago

I've always liked the Mach approach. You've got a few primitives:

- address space

- memory objects

- threads

Mix and match. A Task (process) is not a primitive, but a composite object combining address space with one or more threads. How you fill the address space with actual memory objects is up to you. Map from disk or COW your own address space...have fun!

https://developer.apple.com/library/archive/documentation/Da...

Panzerschrek 1 day ago

The whole approach of using fork seems to be unnatural for me. In many cases (even in the majority of them) it's not needed to inherit the whole structure of the parent process, but to start a given executable. Windows does this better with its CreateProcessW interface.

ComputerGuru 1 day ago

I'm not surprised Chen's patch was rejected; that's an extremely niche usecase not worth supporting. With my shell developer hat on, I agree with the closing "developers would likely welcome a native implementation that isn't (unlike the current implementation) hiding fork() and exec() under the covers".

  • smj-edison 1 day ago

    It sounds like they're interested in the concept though, just not that specific implementation.

codedokode 1 day ago

The problem with replacing exec/fork is that you usually want to configure new process: for example, set up signal handlers, close or open FDs, switch namespaces, setup seccomp, adjust permissions. And all the system calls to do it apply only to the current process and you need something to replace them. The proposal in the article was to create a new API for this.

My idea is that we could make a new syscall, for example "spawn", that creates a new empty process, loads some lightweight "loader" into it, and passes arbitrary configuration data. The loader configures the process and exec()'s the main program. This allows to avoid forking the memory and keep existing APIs, but still requires to fork file descriptors and other things.

  • nyrikki 23 hours ago

    Luckily someone with a time machine saw your post and added it to POSIX.1-2001 :)

    (Sorry if you weren't joking) but yes, posix_spawn() has been a thing and in glibc fork is just a alias to clone()

    Not exactly that OP idea, but fork/exec is legacy really.

trumpdong 23 hours ago

I liked the other proposal where you can create a blank process and then force it to make syscalls, ending with execve. That doesn't require a bunch of special data structures to hold the syscalls you want to do.

tus666 6 hours ago

How can you write for LWN and not have heard of clone(CLONE_THREAD) and multithreading?

asveikau 1 day ago

The things you can do between fork and exec are sometimes underestimated. Off the top of my head, you can call dup2(), you can set a process group id, probably a few other things.

If you contrast that with win32, where you optionally pack a bunch of initial values into a struct, win32 is a much more narrow, less pleasant, less freeform interface, where it is harder to introduce more features.

But I think there is already posix_spawn to imitate that philosophy on Unix-like OSs.

  • loeg 23 hours ago

    > The things you can do between fork and exec are sometimes underestimated. Off the top of my head, you can call dup2(), you can set a process group id, probably a few other things.

    What do you mean underestimated? You can do anything between fork and exec; there are no limitations.

    • dcrazy 23 hours ago

      That’s not true. man 7 signal-safety

      • loeg 21 hours ago

        You're talking about libc design choices, not constraints imposed by the kernel. To the kernel, a post-fork pre-exec process is just any old process. GP was suggesting post-fork processes were constrained in the syscalls they could invoke; they are not.

        • asveikau 20 hours ago

          I did not say they are constrained in what syscalls they can make, as if some nanny at the syscall entry point will punish you for doing wrong. I said that it interacts poorly with threads due to inherent race conditions. See the other comment.

          • loeg 20 hours ago

            > I said that it interacts poorly with threads due to inherent race conditions.

            No, you absolutely did not: https://news.ycombinator.com/item?id=48427396

            Literally nothing in that comment mentions or discusses threads.

            > I did not say they are constrained in what syscalls they can make

            You wrote: "The things you can do between fork and exec are sometimes underestimated. Off the top of my head, you can call dup2(), you can set a process group id, probably a few other things."

            Those are all syscalls. You can also invoke any of the other ~hundreds of syscalls linux exposes, not only dup2, setpgid, and a "few" others.

    • asveikau 23 hours ago

      That's not true. Just one example, if you do anything with threads you are pretty screwed. For example if another thread holds a mutex at the time of fork(2), and you also want that mutex.

      • loeg 21 hours ago

        You can create threads in forked children before exec. Nothing in the kernel prevents you from invoking clone().

        You're talking about libc (glibc) implementation details now; userspace programs running on the Linux kernel do not have to be implemented in C or use glibc's primitives. Your earlier comment I initially replied to was talking about kernel syscalls. Forked processes are free to invoke any syscall they want, not just dup2 or a handful of others.

        • asveikau 21 hours ago

          I'm not talking about glibc implementation details. I'm talking about how mixing fork(2) with threads creates harmful race conditions.

          The forked child has only 1 thread in its process. If the parent's threads are holding a lock or are in the middle of mutating a shared data structure, you're fucked, because those threads are no longer running in your child's copy of the address space and will not finish their work. This issue is fundamental to how threads work and what fork(2) does.

          • loeg 20 hours ago

            Again, you're talking about userspace now. Not kernel-imposed constraints. A userspace program is always free to deadlock itself; fork doesn't change that.

            • asveikau 20 hours ago

              I never said it was a kernel imposed constraint. It remains unsafe behavior, and frankly you'd be stupid to ignore it if you want to write a stable multi threaded program. In colloquial shorthand, you can't do it.

              Signal safety is not the same as this, but similar. I believe posix specifies what is signal-unsafe to be overly broad. But the unsafety isn't an illusion -- it's an emergent property from something being a bad idea given the primitives at work, there are broad categories of bugs that are easy to introduce due to the way it works. So for signals, posix declares a bunch of ill advised things to be undefined, and with good reason. This is an analogous scenario.

            • asveikau 19 hours ago

              Just want to come back with a simple example.

              This means if the program is multi threaded, you cannot rely on calling malloc in the child, because at the time of the fork another thread could have happened to be inside malloc doing manipulations on the global heap.

              Which means, practically speaking, "don't allocate memory between fork and exec".

              If you want to be overly literal as you have been, you can call mmap and it will give you new pages, but who is really doing that? Not the random shared library code you might want to call into. Hell, even a lot of libc calls malloc.

              Which means it's not safe to do a random library call between fork and exec.

              See where I'm going with this? That's if your program is multi threaded. If it isn't, these things are most likely fine.

  • dcrazy 23 hours ago

    posix_spawn is emulated on Linux, but it is a native syscall on macOS (and possibly other OSes?). As discussed in the linked article, there is interest in changing Linux to adopt this model, where posix_spawn is its own fundamental primitive.

    • asveikau 23 hours ago

      Yeah, I think it is a reasonable transition path or implementation detail for some systems to implement it in userland atop fork(2), and others to natively spawn a new process without copying the old address space.

lokar 1 day ago

This seems unnecessary to me. In the example, the core of git should be a library yo can link so you don't need to run the binary. That would be better in every way.

  • sanderjd 1 day ago

    There are lots of reasons to want to spawn fresh processes, which aren't solved by linking a library.

    • aerzen 1 day ago

      Spawning processes should not be on the hot path of any program.

      • 1718627440 1 day ago

        Why? That's a very useful processing primitive.

        • lokar 1 day ago

          It’s a hack with many disadvantages. Sometimes a hack is the right answer, but the kernel should it add a primitive for it.

          • MBCook 1 day ago

            Should bash link in every program the user might want? Load them up as dynamic libraries?

            • m132 18 hours ago

              Node, Python, PowerShell, and the rest do (almost) just that. launchd and systemd famously strived to remove as much shell from the start up process as possible because it was harming boot times and introducing unpredictability.

              • sanderjd 17 hours ago

                I don't know Node or PowerShell very well, but I'm not sure what you mean by this with respect to Python.

                • m132 3 hours ago

                  CPython doesn't usually create subprocesses unless specifically asked to, it loads Python modules and native extensions into its process. The former is similar (you're still extending an existing process with new code, just interpreted), the latter is literally dlopen(), so loading dynamic libraries.

                  A lot of other Python implementations don't have the ability to spin up new processes at all too.

                  • sanderjd 2 hours ago

                    I still don't really get this point. It's just two different things, spawning processes and running libraries. Seems like you're comparing apples and oranges to me.

            • lokar 3 hours ago

              Bash as an interactive tool is very different. It is used to run an almost arbitrary number of things, and a pretty low rate.

              Bash as a programming language is just a bad idea.

          • sanderjd 21 hours ago

            Aren't we discussing just such a primitive?

            • lokar 3 hours ago

              oh, sorry, typo

              s/it/not/

      • pizlonator 1 day ago

        It ends up on the hot path of programs that use process isolation aggressively

        • lokar 3 hours ago

          Sure, and there a primary thing you want is a whole new environment/context for the child (new environment, fds, memory, cgroups, namespaces, etc).

    • lokar 1 day ago

      Sure, but not many times a second

      • kllrnohj 1 day ago

        Every build system ever says hello.

      • sanderjd 21 hours ago

        Why not?

        • m132 18 hours ago

          Because it comes with a lot of overhead and, unless for some reason you really need every of those processes to have their own address space, set of privileges, file descriptors, etc., there's no point in wasting resources repeatedly setting those up only to tear them down milliseconds later. Running the same workloads in an nginx-style process pool usually works better.

          • sanderjd 17 hours ago

            I see what you mean now. I agree, a sustained workload of creating many processes very quickly is probably not a great idea. But it's also useful to be able to spawn that process pool (and any number of other use cases like that) efficiently.

  • 1718627440 1 day ago

    But when you use a process, you get tons of things for free, the subtask is invoked in parallel, you get isolation and you can control execution for free. Unless you are already writing a multithreaded program or already accept passing objects in memory, using a process is actually easier to write than using a library.

    If I use a library, I also need to start using threads and need to invent some core synchronization mechanism. I essentially are reinventing a small scheduler, when I already get this from the OS for free. Also know any crash in the third-party code will crash the whole program, the third-party code has access to the whole address space. With invoking a process you also have a standardized API implemented by the OS.

    • lokar 21 hours ago

      I'm not sure what you mean by inventing a sync mechanism, all languages come with one. Same with a scheduler, either your language runtime or the OS (or both) will deal with scheduling.

  • omoikane 1 day ago

    Launching git repeatedly was probably not the best example. But it's hard to think of good examples where launching processes repeatedly is the most performant thing to do, probably because launching processes had been expensive and everyone has learned to do something else (libraries, zygotes, etc). Maybe a different question is: if launching processes were cheap, is there something we would implement as processes instead of libraries?

    I can recall just one program that's intentionally not implemented as a library, but I think people have since built a library on top of it:

    https://dechifro.org/dcraw/#:~:text=Why%20don%27t%20you%20im...

debatem1 1 day ago

There are a lot of slightly different fork-exec-like things in the concept space and it's hard to imagine one approach satisfying them all. IMO it would be interesting to take an approach analogous-ish to sched_ext_ops where you built the rough flow chart of a combined fork-exec, but with hooks built to enable ebpf to change behavior or skip the bits these sophisticated users don't want/need.

  • MBCook 1 day ago

    Fork/exec is great if you actually want the traditional copy of your process for some reason.

    For launching something totally new, like the example in the article of some tool calling git, I think it does make a ton of sense to make something new.

    Especially since I suspect that is by far the more common case. I suspect “I want a clone of me“ is relatively rarely used at this point.

    • debatem1 20 hours ago

      Relatively rarely, but in some performance sensitive use cases. Mine happens to be fuzzers, where a very cheap fork-like primitive would be a really big win.

      • surajrmal 15 hours ago

        Android and chrome both benefit greatly from fork exec as part of their zygote model iirc. It substantially reduces the memory cost and latency of spawning new apps and tabs.

foo-bar-baz529 14 hours ago

This isn’t moving beyond fork and exec at all. It’s adding a complicated API for a marginal gain for a niche use case, and ignoring the actual big bottleneck of fork

mike_hock 1 day ago

The most astonishing part is that this is dated June 5th, 2026.

I.e. a year that starts with 20, not 19.

  • JdeBP 1 day ago

    These discussions were definitely had back in the 20th century too. The spawn model versus the fork+execve model has been an on-going debate since the time of MS/PC/DR-DOS.

Sophira 1 day ago

I'm guessing that a big part of the problem with moving away from fork() in general is that each new process needs a copy of the parent process' environment anyway, right?

  • lokar 1 day ago

    the environment is not that big

  • dijit 1 day ago

    I'm a bit naive, but I don't think that's necessarily a requirement.

    It might be commonly held convention, and thus, an assumption, in Linux (and, broadly, UNIX) but I don't think it's true inside VAX or even Windows, so I don't think it's a requirement.

    Unless I've missed something (which is totally possible, this is not an area of OS design I've spent much time).

    • sjmulder 1 day ago

      Even DOS has environment inheritance!

    • lanstin 1 day ago

      But also UID, groups, controlling TTY, process group, capabilities, pipes, shared memory, etc. and the file descriptors while maybe not inherently needed are how a lot of Unix plumbing works.

  • zerobees 1 day ago

    The LWN article is incorrect in saying that it "must copy the entire process state (including memory) for the child process". There are some kernel structures and page tables that need to be initialized, plus you need a new stack, but it's not nearly as dramatic as implied. Most of the parent's memory is "incorporated by reference", so to speak.

    In fact, if you profile it, in the fork() + execve() model, execve() is far more expensive, because not only does it replace the old process with a new one, but it also involves running the dynamic linker, which opens, parses, and mmaps library files.

    It still makes sense to get rid of the fork() overhead if you're going to throw away the cloned process state soon thereafter, but if you wanted to make process execution radically faster, rethinking the exec architecture would probably offer more significant gains.

    • nasretdinov 1 day ago

      Fork becomes more and more expensive the higher the RSS of the process, roughly 1ms per 1Gb of the process size with 4kb pages. Given that modern servers can easily support 1-2Tb of RAM the fork() part can easily take several hundred milliseconds, blocking everything in the meantime. So for larger programs you kinda have to have a "fork helper" process if you need to execute external programs for some reason.

    • corbet 1 day ago

      The kernel does not copy every page, but it does have to copy all of the VMAs. Setting memory to COW (which can involve changing a lot of page-table-entries) is not free either. I guess I could have mentioned copy-on-write explicitly, but I do not believe that what I wrote was incorrect.

  • sanderjd 1 day ago

    A lot of times you actively don't want the parent environment or any of the memory or file descriptors. And then you have to actively do work to fix all that stuff up after the fork.

ggm 1 day ago

Aesthetically I have no intention of moving beyond. I'm content with my kernels scheduler and how it maps "heavyweight" processes to cores.

I do use threaded code. It's significantly harder to write and reason about. (45 years in to a CS career, ageing out)

You have to be clever to do better than clever people. Clever people bootstrapped me into fork()/exec() and I know my limits.

  • redleader55 1 day ago

    When cores start needing more than 9 bits to be represented and RAM is in terabytes, many of the old assumptions need to change. Schedulers need to be implemented in userspace, RAM needs to be allocated in GB, not in 4k, io needs to require less round-trips between kernel and user space and NICs need to do a lot more work before the data reaches the CPU.

    • skydhash 23 hours ago

      Does it need to be the same OS? Most consumer device are in the low 16GB range for memory with some outliers in the 64 and 128 GB. 32 cores are still in the realm of specialized devices.

      Yes, we’re not the one paying for Linux development, but its subsystems are so complicated for general purpose computing. Like fitting formula 1 car parts onto a camry.

      • tadfisher 23 hours ago

        Our software is littered with the consequences of these kinds of assumptions, and they have an impact on consumer use cases.

        x86 still runs in real mode on boot despite dropping the PC BIOS.

        Lots of software still assumes a 4kb page size, to the point where migrating Android to 16kb is an ongoing multi-year effort involving far too many people. And this is an OS for phones, which you might assume would lack the memory to benefit from a larger page size.

        And one of the most popular consumer CPUs for enthusiasts, the Ryzen X3D chips, broke assumptions in both Linux and Windows schedulers that all cores have access to the same amount of L3 cache.

        I would probably not assume the kinds of hardware limitations that we have now will persist into the useful lifetime of current software. Splitting the OS into "consumer" and "enterprise" variants is one of those moves that would bake in a ton of assumptions and make things messier in the future.

        • skydhash 22 hours ago

          It’s all about contracts. It’s fine to define assumptions and build software on top of those. It’s also fine to break those and adjust the software. The trap is trying to steer towards a universal solution (Yagni is the cure there) or trying to slip something in that does not respect the contracts (hence bugs).

          UEFI could have supported something like ELF and do away with real mode. Intel and Amd could have just introduced a new line of cpu and everyone could have transitioned to that (with maybe shims to soften the change). But everyone is all about backwards compatibility and compile once, runs for eternity.

  • skydhash 1 day ago

    I’m using Emacs and various cli tools and while threads are nice to have, they can easily ramp up the complexity of a program beyond what is necessary. I much prefer the boilerplate of setting up a thread pool and tasks queue, rather than dealing with all the await/async syntactic sugar.

stevefan1999 18 hours ago

If fork and exec can exhibit persistent and algebraic behavior (beyond its CoW nature) that would not only be more useful but more interesting to use, for example using it for doing lazy evaluation

a-dub 1 day ago

i thought this was all fixed with special modes of clone that are optimized and don't actually copy anything (ie, it creates a new deficient process that can pretty much only exec)?

LoganDark 17 hours ago

Huh, LWN has moved to (sometimes) requiring a click to proceed past the subscription pitch to the actual article. I feel like this may have an inverse effect (insistent begging to the point of inserting additional obstacles = angry/insulted users that are less likely to pay).

burnt-resistor 1 day ago

> "If you are repeatedly creating large processes, you are already doing it wrong. The fix is in user space, not the kernel."

Every couple of years, someone claims they have "the solution" implying everyone else who came before them didn't know what they were doing.

  • yxhuvud 1 day ago

    It can also mean that neither the hardware side or the software side is static, but change over time. That means that their demands and what they allow also change over time. This leads to the insight that what was perhaps a good idea on 70s hardware/software is not necessarily a good, or even ok, idea 50 years later on modern hardware executing OSes and programs that have been kept up to date.

hparadiz 1 day ago

Maybe tangentially related but I always think it's silly that every linux process has the same libgcc_so.so.1 loaded into memory for each process even though the raw binary for the library is exactly the same so you end up with like 800 copies of libgcc_so.so.1 in memory.

I mean maybe this has been optimized for already and I don't know what I'm talking about but maybe someone with more knowledge about the kernel knows? Is this something we simply can't optimize for because of security implications?

  • 201984 1 day ago

    Shared libraries (and mmapped files in general) are deduplicated; it's nowhere near as bad as you think. The kernel loads a .so into memory once and then maps that memory into every process that mmaps it.

    Editing to add: this deduplication is one of the greatest upsides to dynamic linking. Common libs like libgcc and libc only have to exist in memory once and can stay in CPU caches, whereas if they were statically linked into every binary, each binary would have a copy of that library that wouldn't be shared with anything else and you'd waste a lot of memory.

    • sjmulder 1 day ago

      Doesn't the loaded code have to be patched for relocations?

      • ptspts 1 day ago

        It does, so not 100% is reused. The patched parts are in different sections though, so the entire .text (code) section ends up being reused.

      • monocasa 1 day ago

        Not on modern archs that provide decent support for PIE (position independent executables).

        • 201984 1 day ago

          How do you think position independent code can call functions from other .so's without being patched with their addresses?

          They can't, so even PIC code still has to have a relocation table that gets patched. It's in a different page than the code though, so code does still get reused.

          • monocasa 1 day ago

            That's not really patching though, any more than any use of function pointers is patching.

            • 201984 23 hours ago

              There's a part of the .so ELF file (the Global Offset Table aka GOT) that has to be modified with all the addresses of the functions being imported, which of course vary from process to process.

              If not patching, what exactly would you call modifying part of the file?

              • monocasa 22 hours ago

                And the got is just a big table of pointers like any other table of pointers your application manipulates as it runs.

                This isn't meant as a reductive take, but instead that there is a difference between completely describable in C like the contents of the .got section, and something like a .reloc section that actually has to understand the generated assembly in order to build the relocation table to load and link the executable. Both are linking, but I've saved "patching" for more brain surgery esque techniques. Like on mips, the jump instruction immediate is the bottom 26 bits of the absolute address of the target, so you're going through and modifying all of the jump instructions if you load it to somewhere it wasn't linked at.

      • t-3 1 day ago

        Not if it's position-independent.

  • monocasa 1 day ago

    Those mappings by default all go to the same shared memory.

    Unices have been sharing executable memory between processes longer than there's been mmap for user space to do the same thing themselves. I remember seeing it in the 2BSD kernel for instance.

  • saidinesh5 1 day ago

    Typically libgcc_so.so is loaded by the linker, which uses an mmap call to map the binary into the address space.

    > The kernel keeps track of which file is mapped where, and can detect when a request is made to map an already mapped file again, avoiding physical memory allocation if possible.

    Relevant stack overflow answer: https://stackoverflow.com/questions/61950951/linux-shared-li...

  • mlaretallack 1 day ago

    In Linux, when a shared lib is loaded by multiple processes, its loaded once and not duplicated in ram. Only if a memory page is modified by the process will the memory be duplicated. (Hope I have explained that correctly)

  • BoingBoomTschak 1 day ago

    Eh? Aren't shared libraries actually shared in memory?

    • 1718627440 1 day ago

      Yeah, that's kind of the point.

      • johnthescott 21 hours ago

        shared libraries does not imply shared in ram only.

  • sirsinsalot 1 day ago

    I have a rule for myself. If I think something is silly or stupid, I assume I don't understand it. I usually find I do not understand it, and it no longer seems silly when I do understand it.

    In this case too, you think it is silly because you don't understand it. Your assumptions are wrong, making it seem silly.