Lots of in-line reply-bits (;-)) On Tuesday, December 26, 2017 at 8:58:47 AM UTC-5, Fabien wrote: > > Thank you, that's a very interesting read. > > TLDR: If I understood everything, NP-complete != DLL hell, but anyway, go > is vulnerable to DLL hell, so we're screwed. > > I'm not sure I understand everything, though. Both your article and Russ > Cox' you mention seem to confuse NP-completeness and the concept of DLL > hell. I thought DLL hell was the inability, plain and simple, to install > two versions of the same DLL (and, AFAIK, two different versions of a go > package). The problem of whether you need two incompatible versions or not > happens to be NP-complete, but that's not the problem per se, is it ? >
I view the inability to have two versions of the same thing as the cause, and DLL hell the effect, with finding a solution to DLL Hell NP-hard. There is a weaker sense of DLL hell too, that of having to find all the different dependancies. We started out with things like "rpm search", but folks quickly automated. I stil lremember trying to find all the dependancies for a copy of MythTV. Erk! You also see people adressing that weaker case via flatpacks, docker images and the like, and it somewhat mitigates the harder case. > If I understood correctly your article and Russ Cox', the proposed > solution is either to offer, in each subsequent version of a package / > shared library / DLL, both the old and the new API, or to allow the > installation of several packages. However this leads to problems, as you > now tend to have bloated binaries that contain old compatibility code you > don't necessarily need. If version 1 and 2 of package C are incompatible, > I'm expecting the latest binary version of package C to contain both > version 1 and version 2. > Yes: we're trading space for time. In Multics there were a number of things you could do with the binder to keep the RSS small, at the cost of a larger on-dicklibrary image. > > However that's good for DLLs and other shared objects, but that seems > overkill in the case of go, since each executable includes all the library > code it uses. > In a static-linking environment like Go or Linux, you can drop all the dead code when you create the final executable. There is a recent article at LWN about doing so for Linux at https://lwn.net/SubscriberLink/741494/2efc53db170f03c5/ > If I need package A and package B, and both those packages only use the > version 2 of C, I don't wan't to include the version 1 of C, too, in my > executable: it is already big enough. If you want to avoid it, you need to > be able to specify specific versions of packages, and you're almost back to > square one, with an NP-hard problem. > But this is not DLL hell anymore: first, you know your program can be > installed, not matter what. You might need to duplicate lots of code, have > several different versions of your packages, but at least there's a > possible solution. Secondly, this is no more a decision problem (is there a > solution at all), but an optimization problem (what is the best possible > solution): the initial, suboptimal solution (just install the last version > of each package, recursively) is found in linear time and the optimization > process can be interrupted at any moment if it takes too long, leaving you > with a suboptimal but workable solution. > > So, for instance : > > let's say I want to install package p, that requires both packages a and b. > > Here are package a's versions, and their dependencies : > > a: > v1: c >=1 > v2: c = 2 > > And package b: > > b: > v1: c >= 2, d >= 1 > v2: c >= 3, d >= 1 > > There are two more dependencies, c and d, here are their own versions and > dependencies: > > c: > v1: nothing > v2: nothing > v3: nothing > > d: > v1: c = 1 > > If I can't have two versions of the same package installed, I have to go > through a NP-complete problem (and use a SAT solver or equivalent) to find > I'm actually out of luck: there is now way to install p while having at > most one version of each package. That is DLL hell. > > But if I can have several versions installed, a first reasonable option is > to propose to install the latest version of a and b, and then the latest > versions of each of their dependencies, recursively. If incompatible > versions are needed, duplicate them. So, here, that initial solution would > be a2, b2, c1, c2, c3, d1. Good, and found in linear time, but not perfect, > since there are duplicate packages. I can now try to find an optimal > solution, optimal being defined as "the newer versions the better, the > least installed packages the better". Here, the optimal solution is a1, b2, > c1, c3, d1 which is only 4 packages, the trick being to downgrade to > version 1 of package a. Still NP-complete (NP hard, actually), but no DLL > hell, and I already had a workable solution in linear time. > There is a longish discusion of David J (my evil twin) Brown's approach to latest-version-preference in https://www.usenix.org/legacy/publications/library/proceedings/als00/2000papers/papers/full_papers/browndavid/browndavid_html/ We were woking in a dynamic environment, so actually droppping the unneeded bits wasn'ty part of the plan. We probaqbly could have page-aligned a lot of stuff in the libraries, too, so that we reduce the RSS. No-one has discussed the garbage elimination problem, though. In development, we only needed to ship and support a few versions, as intermediate versions were unused by anyone save I and my colleagues (Hey, Edsel!). Arguably we need a way to phase out old versions in GLIBC and the kernel. > Now, am I wrong to think that go, in its current state, is still > vulnerable to DLL hell, even with the vendoring mechanism? > Yes, but the likliehood is reduced, and the pain is localized to the point at which you try to move forwared to a new release of a dependency. You either suceed or you defer the update until the dependanct's dependancies are up to date. I'm biased: I think the GLIBC approach should be used universally throughout Linux so that I can avoid reimplementing it in Go, node.js, Rust, Python, Perl, ... ite ad infinitum (;-)) --dave -- You received this message because you are subscribed to the Google Groups "golang-nuts" group. To unsubscribe from this group and stop receiving emails from it, send an email to golang-nuts+unsubscr...@googlegroups.com. For more options, visit https://groups.google.com/d/optout.