On Sun, Mar 24, 2019 at 9:51 PM Saleem Abdulrasool <compnerd at compnerd.org> wrote:> Sorry for the late chiming in. > > Yes, swift does use autolinking, and I would like to use that on all the > targets. The only target which does not support this functionality > currently are ELF based. That said, I think that `#pragma comment(link, > ...)` is insufficient for my needs. Building Foundation requires framework > style linking as well. The original design that I had in mind was derived > from ld64 and link. Personally, I still strongly favour link's behaviour > of parsing "command line" options from the object files when they are > loaded. There was strong opposition to that approach from Rui though. > Would we want to have special pragmas for each "feature"? >I think as you double-quoted the words "command line", I believe we are on the same page, but what I opposed was to allow arbitrary command line options in the .linker-options section. That is really a can of worms that we don't want to open. I didn't oppose allowing a limited set of command-line-ish directives. That said I'd prefer the proposal of this thread over the .linker-options though. The ELF model doesn't have the simplistic model for processing the command> line that PE/COFF does. Because ordering is relevant to the model, it > would be ideal to process them inline, but, since lld already moves far > enough away from the traditional Unix model, perhaps we can simplify it to > append the command line directives to the end of the command line. > > The other case that is interesting to think about is the autolinking > support in C++ (and clang) modules. > > On Thu, Mar 21, 2019 at 9:49 AM bd1976 llvm <bd1976llvm at gmail.com> wrote: > >> On Thu, Mar 21, 2019 at 12:06 AM Rui Ueyama <ruiu at google.com> wrote: >> >>> Perhaps there's no one clean way to solve this issue, because previously >>> all libraries and object files are explicitly given to the linker via a >>> command line and the order of files in the command line matters. That >>> assumes human intervention to work correctly. Now, the autolinking feature >>> will add libraries implicitly. Since it's implicit, there will be only one >>> way how that works, so sometimes that works and sometimes doesn't. >>> >>> It feels to me that we should aim for making it work reasonably well for >>> reasonable use cases. By reasonable use cases, I'm thinking of the >>> following: >>> >>> 1. --static option may or may not be given (i.e. we should allow that >>> feature for both static linking and dynamic linking.) >>> 2. There are no competing defined symbols in a given set of libraries, >>> or if they exist, the program owner doesn't care which is linked to their >>> program. >>> 3. There may be circular dependencies between libraries. >>> >>> I don't think the above assumption is too odd. If I have to implement >>> the autolinking feature to GNU linker for the above scenario, I'd probably >>> use the following scheme: >>> >>> 1. While reading object files, memorize libraries that are autolinked >>> 2. After linking everything, create a list of files consisting of >>> autolinked libraries AND libraries given via the command line >>> 3. Visit each file in the list as if they were wrapped in --start-group >>> and --end-group. >>> >>> I'd think the above scheme should work reasonably well. What do you >>> think? >>> >> >> Very nice. I agree with your definition of "reasonable" usecaes >> (actually, as I have said before, I think that restricting autolinking to >> this "reasonable" set is actually a feature - to avoid developers having >> source code that only works with a particular linker). I also like the >> proposal for a GNU implementation - I think this is enough to show that >> GNU-like linkers could implement this. >> >> At this point I will try to prototype this up so that people have an >> implementation to play with. >> >> I am keen to hear from Saleem (compnerd) on this, as he did the original >> .linker-options work. >> >> >>> >>> On Tue, Mar 19, 2019 at 11:02 AM bd1976 llvm <bd1976llvm at gmail.com> >>> wrote: >>> >>>> On Mon, Mar 18, 2019 at 8:02 PM Rui Ueyama <ruiu at google.com> wrote: >>>> >>>>> On Thu, Mar 14, 2019 at 1:05 PM bd1976 llvm via llvm-dev < >>>>> llvm-dev at lists.llvm.org> wrote: >>>>> >>>>>> On Thu, Mar 14, 2019 at 6:27 PM Peter Collingbourne <peter at pcc.me.uk> >>>>>> wrote: >>>>>> >>>>>>> >>>>>>> >>>>>>> On Thu, Mar 14, 2019 at 6:08 AM bd1976 llvm via llvm-dev < >>>>>>> llvm-dev at lists.llvm.org> wrote: >>>>>>> >>>>>>>> At Sony we offer autolinking as a feature in our ELF toolchain. We >>>>>>>> would like to see full support for this feature upstream as there is >>>>>>>> anecdotal evidence that it would find use beyond Sony. >>>>>>>> >>>>>>>> In general autolinking (https://en.wikipedia.org/wiki/Auto-linking) >>>>>>>> allows developers to specify inputs to the linker in their source code. >>>>>>>> LLVM and Clang already have support for autolinking on ELF via embedding >>>>>>>> strings, which specify linker behavior, into a .linker-options section in >>>>>>>> relocatable object files, see: >>>>>>>> >>>>>>>> RFC - >>>>>>>> http://lists.llvm.org/pipermail/llvm-dev/2018-January/120101.html >>>>>>>> LLVM - >>>>>>>> https://llvm.org/docs/Extensions.html#linker-options-section-linker-options, >>>>>>>> https://reviews.llvm.org/D40849 >>>>>>>> Clang - >>>>>>>> https://clang.llvm.org/docs/LanguageExtensions.html#specifying-linker-options-on-elf-targets, >>>>>>>> https://reviews.llvm.org/D42758 >>>>>>>> >>>>>>>> However, although support was added to Clang and LLVM, no support >>>>>>>> has been implemented in LLD; and, I get the sense, from reading the >>>>>>>> reviews, that there wasn't agreement on the implementation when the changes >>>>>>>> landed. The original motivation seems to have been to remove the >>>>>>>> "autolink-extract" mechanism used by Swift to workaround the lack of >>>>>>>> autolinking support for ELF. However, looking at the Swift source code, >>>>>>>> Swift still seems to be using the "autolink-extract" method. >>>>>>>> >>>>>>>> So my first question: Are there any users of the current >>>>>>>> implementation for ELF? >>>>>>>> >>>>>>>> Assuming that no one is using the current code, I would like to >>>>>>>> suggest a different mechanism for autolinking. >>>>>>>> >>>>>>>> For ELF we need limited autolinking support. Specifically, we only >>>>>>>> need support for "comment lib" pragmas ( >>>>>>>> https://docs.microsoft.com/en-us/cpp/preprocessor/comment-c-cpp?view=vs-2017) >>>>>>>> in C/C++ e.g. #pragma comment(lib, "foo"). My suggestion that we keep the >>>>>>>> implementation as lean as possible. >>>>>>>> >>>>>>>> Principles to guide the implementation: >>>>>>>> - Developers should be able to easily understand autolinking >>>>>>>> behavior. >>>>>>>> - Developers should be able to override autolinking from the linker >>>>>>>> command line. >>>>>>>> - Inputs specified via pragmas should be handled in a general way >>>>>>>> to allow the same source code to work in different environments. >>>>>>>> >>>>>>>> I would like to propose that we focus on autolinking exclusively >>>>>>>> and that we divorce the implementation from the idea of "linker options" >>>>>>>> which, by nature, would tie source code to the vagaries of particular >>>>>>>> linkers. I don't see much value in supporting other linker operations so I >>>>>>>> suggest that the binary representation be a mergable string section >>>>>>>> (SHF_MERGE, SHF_STRINGS), called .autolink, with custom type >>>>>>>> SHT_LLVM_AUTOLINK (0x6fff4c04), and SHF_EXCLUDE set (to avoid the contents >>>>>>>> appearing in the output). The compiler can form this section by >>>>>>>> concatenating the arguments of the "comment lib" pragmas in the order they >>>>>>>> are encountered. Partial (-r, -Ur) links can be handled by concatenating >>>>>>>> .autolink sections with the normal mergeable string section rules. The >>>>>>>> current .linker-options can remain (or be removed); but, "comment lib" >>>>>>>> pragmas for ELF should be lowered to .autolink not to .linker-options. This >>>>>>>> makes sense as there is no linker option that "comment lib" pragmas map >>>>>>>> directly to. As an example, #pragma comment(lib, "foo") would result in: >>>>>>>> >>>>>>>> .section ".autolink","eMS", at llvm_autolink,1 >>>>>>>> .asciz "foo" >>>>>>>> >>>>>>>> For LTO, equivalent information to the contents of a the .autolink >>>>>>>> section will be written to the IRSymtab so that it is available to the >>>>>>>> linker for symbol resolution. >>>>>>>> >>>>>>>> The linker will process the .autolink strings in the following way: >>>>>>>> >>>>>>>> 1. Inputs from the .autolink sections of a relocatable object file >>>>>>>> are added when the linker decides to include that file (which could itself >>>>>>>> be in a library) in the link. Autolinked inputs behave as if they were >>>>>>>> appended to the command line as a group after all other options. As a >>>>>>>> consequence the set of autolinked libraries are searched last to resolve >>>>>>>> symbols. >>>>>>>> >>>>>>> >>>>>>> If we want this to be compatible with GNU linkers, doesn't the >>>>>>> autolinked input need to appear at the point immediately after the object >>>>>>> file appears in the link? I'm imagining the case where you have a >>>>>>> statically linked libc as well as a libbar.a autolinked from a foo.o. The >>>>>>> link command line would look like this: >>>>>>> >>>>>>> ld foo.o -lc >>>>>>> >>>>>>> Now foo.o autolinks against bar. The command line becomes: >>>>>>> >>>>>>> ld foo.o -lc -lbar >>>>>>> >>>>>> >>>>>> Actually, I was thinking that on a GNU linker the command line would >>>>>> become "ld foo.o -lc -( -lbar )-"; but, this doesn't affect your point. >>>>>> >>>>>> >>>>>>> >>>>>>> If libbar.a requires an additional object file from libc.a, it will >>>>>>> not be added to the link. >>>>>>> >>>>>>> >>>>>> As it stands all the dependencies of an autolinked library must >>>>>> themselves be autolinked. I had imagined that this is a reasonable >>>>>> limitation. If not we need another scheme. I try to think about some >>>>>> motivating examples for this. >>>>>> >>>>>> >>>>>>> 2. It is an error if a file cannot be found for a given string. >>>>>>>> 3. Any command line options in effect at the end of the command >>>>>>>> line parsing apply to autolinked inputs, e.g. --whole-archive. >>>>>>>> 4. Duplicate autolinked inputs are ignored. >>>>>>>> >>>>>>> >>>>>>> This seems like it would work in GNU linkers, as long as the >>>>>>> autolinked file is added to the link immediately after the last mention, >>>>>>> rather than the first. Otherwise a command line like: >>>>>>> >>>>>>> ld foo1.o foo2.o >>>>>>> >>>>>>> (where foo1.o and foo2.o both autolink bar) could end up looking >>>>>>> like: >>>>>>> >>>>>>> ld foo1.o -lbar foo2.o >>>>>>> >>>>>>> and you will not link anything from libbar.a that only foo2.o >>>>>>> requires. It may end up being simpler to not ignore duplicates. >>>>>>> >>>>>> >>>>>> Correct; but, given that the proposal was to handle the libraries as >>>>>> if they are appended to the link line after everything on the command line >>>>>> then I think this will work. With deduplication (and the use of SHF_MERGE) >>>>>> developers get no ordering guarantees. I claim that this is a feature! My >>>>>> rationale is that the order in which libraries are linked affects different >>>>>> linkers in different ways (e.g. LLD does not resolve symbols from archives >>>>>> in a compatible manner with either the Microsoft linker or the GNU >>>>>> linkers.), by not allowing the user to control the order I am essentially >>>>>> saying that autolinking is not suitable for libraries that offer competing >>>>>> copies of the same symbol. This ties into my argument that "comment lib" >>>>>> pragmas should be handled in as "general" a way as possible. >>>>>> >>>>> >>>>> Right. I think if you need a fine control over the link order, >>>>> autolinking is not a feature you want to use. Or, in general, if your >>>>> program is sensitive to a link order because its source object files have >>>>> competing symbols of the same name, it's perhaps unnecessarily fragile. >>>>> >>>>> That being said, I think you need to address the issue that pcc >>>>> pointed out. If you statically link a program `foo` with the following >>>>> command line >>>>> >>>>> ld -o foo foo.o -lc >>>>> >>>>> , `foo.o` auto-imports libbar.a, and libbar.a depends on libc.a, can >>>>> your proposed feature pull out object files needed for libbar.a? >>>>> >>>> >>>> It won't work on GNU linkers. It will work with LLD as LLD has >>>> MSVC-like archive handling. However, I would like to make sure that >>>> whatever we come up with can be supported in the GNU toolchain. >>>> >>>> I had thought that it would be acceptable that all the dependencies of >>>> an autolinked library must themselves be autolinked in order to work on GNU >>>> style linkers. Having thought more, I don't like this limitation - >>>> especially as it doesn't exist for Microsoft style linkers. One possible >>>> resolution could be that GNU linkers might have to implement another >>>> command line option e.g. --auto-dep=<file> to allow injection into the >>>> group of autolinked libraries. >>>> >>>> i.e In pcc's example you would need to do: "ld foo.o --auto-dep=libc.a" >>>> which would become "ld --start-group libbar.a libc.a --end-group" with >>>> autolinking. >>>> >>>> I wanted to avoid the approach of inserting autolinked libraries after >>>> the object that autolinks them. In LLD (and MSVC) it becomes hard to reason >>>> about "where" the linker is in the command line and it would also mean that >>>> we can't have the nice separation between parsing the command line and >>>> doing the rest of the link that we currently have. Also, if you give people >>>> a way to have a fine grained control over the link order with autolinking >>>> you risk ending up with source code that will link on GNU style linkers but >>>> not with LLD (assuming GNU ever implemented support for autolinking). >>>> >>>> Scenario: >>>> >>>> libbar.a(bar.o) - defines symbol bar >>>> libfoo.a(foo.o) - defines foo and autolinks libbar.a >>>> main.o - references foo >>>> another.o - does not reference foo >>>> No references to bar exist >>>> >>>> lld -lfoo another.o --whole-archive main.o with autolinking becomes lld >>>> -lfoo another.o --whole-archive main.o -lbar result: bar.o gets added to >>>> the link. >>>> But, if a change is made so that another.o references bar then the link >>>> line with autolinking becomes lld -lfoo another.o -lbar --whole-archive >>>> main.o result: bar.o is not added to the link. >>>> >>>> Hopefully the above scenario demonstrates why I think that it becomes >>>> too complicated to reason about the effects of autolinking with pcc's >>>> proposed insertion scheme. >>>> >>>> >>>> >>>>> 5. The linker tries to add a library or relocatable object file from >>>>>>>> each of the strings in a .autolink section by; first, handling the string >>>>>>>> as if it was specified on the commandline; second, by looking for the >>>>>>>> string in each of the library search paths in turn; third, by looking for a >>>>>>>> lib<string>.a or lib<string>.so (depending on the current mode of the >>>>>>>> linker) in each of the library search paths. >>>>>>>> >>>>>>> >>>>>>> Is the second part necessary? "-l:foo" causes the linker to search >>>>>>> for a file named "foo" in the library search path, so it seems that >>>>>>> allowing the autolink string to look like ":foo" would satisfy this use >>>>>>> case. >>>>>>> >>>>>> >>>>>> >>>>>> I worded the proposal to avoid mapping "comment lib" pragmas to >>>>>> --library command line options. My reasons: >>>>>> >>>>>> 1. I find the requirement that the user put ':' in their lib strings >>>>>> slightly awkward. It means that the source code is now coupled to a >>>>>> GNU-style linker. So then this isn't merely an ELF linking proposal, it's a >>>>>> proposal for ELF toolchains with GNU-like linkers (e.g. the arm linker >>>>>> doesn't support the colon prefix >>>>>> http://infocenter.arm.com/help/index.jsp?topic=/com.arm.doc.dui0474c/Cjahbdei.html >>>>>> ). >>>>>> >>>>>> 2. The syntax is #pragma comment(lib, ...) not #pragma >>>>>> linker-option(library, ...) i.e. the only thing this (frankly rather >>>>>> bizarre) syntax definitely implies is that the argument is related to >>>>>> libraries (and comments ¯\_(ツ)_/¯); it is a bit of a stretch to interpret >>>>>> "comment lib" pragmas as mapping directly to "specifying an additional >>>>>> --library command line option". >>>>>> >>>>>> AFAIK all linkers support two ways of specifying inputs; firstly, >>>>>> directly on the command line; secondly, with an option with very similar >>>>>> semantics to GNU's --library option. I choose a method of finding a input >>>>>> files that encompasses both methods of specifying a library on the command >>>>>> line. I think that this method is actually more intuitive than either the >>>>>> method used by the linker script INPUT command or by --library. FWIW, I >>>>>> looked into the history of the colon prefix. It was added in >>>>>> https://www.sourceware.org/ml/binutils/2007-03/msg00421.html. >>>>>> Unfortunately, the rationale given is that it was merely a port of a >>>>>> vxworks linker extension. I couldn't trace the history any further than >>>>>> that to find the actual design discussion. The linker script command INPUT >>>>>> uses a different scheme and the command already had this search order 20 >>>>>> years ago, which is the earliest version of the GNU linker I have history >>>>>> for; again, the rationale is not available. >>>>>> >>>>>> >>>>>>> 6. A new command line option --no-llvm-autolink will tell LLD to >>>>>>>> ignore the .autolink sections. >>>>>>>> >>>>>>>> Rationale for the above points: >>>>>>>> >>>>>>>> 1. Adding the autolinked inputs last makes the process simple to >>>>>>>> understand from a developers perspective. All linkers are able to implement >>>>>>>> this scheme. >>>>>>>> 2. Error-ing for libraries that are not found seems like better >>>>>>>> behavior than failing the link during symbol resolution. >>>>>>>> 3. It seems useful for the user to be able to apply command line >>>>>>>> options which will affect all of the autolinked input files. There is a >>>>>>>> potential problem of surprise for developers, who might not realize that >>>>>>>> these options would apply to the "invisible" autolinked input files; >>>>>>>> however, despite the potential for surprise, this is easy for developers to >>>>>>>> reason about and gives developers the control that they may require. >>>>>>>> 4. Unlike on the command line it is probably easy to include the >>>>>>>> same input file twice via pragmas and might be a pain to fix; think of >>>>>>>> Third-party libraries supplied as binaries. >>>>>>>> 5. This algorithm takes into account all of the different ways that >>>>>>>> ELF linkers find input files. The different search methods are tried by the >>>>>>>> linker in most obvious to least obvious order. >>>>>>>> 6. I considered adding finer grained control over which .autolink >>>>>>>> inputs were ignored (e.g. MSVC has /nodefaultlib:<library>); however, I >>>>>>>> concluded that this is not necessary: if finer control is required >>>>>>>> developers can recreate the same effect autolinking would have had using >>>>>>>> command line options. >>>>>>>> >>>>>>>> Thoughts? >>>>>>>> >>>>>>>> _______________________________________________ >>>>>>>> LLVM Developers mailing list >>>>>>>> llvm-dev at lists.llvm.org >>>>>>>> https://lists.llvm.org/cgi-bin/mailman/listinfo/llvm-dev >>>>>>>> >>>>>>> >>>>>>> >>>>>>> -- >>>>>>> -- >>>>>>> Peter >>>>>>> >>>>>> _______________________________________________ >>>>>> LLVM Developers mailing list >>>>>> llvm-dev at lists.llvm.org >>>>>> https://lists.llvm.org/cgi-bin/mailman/listinfo/llvm-dev >>>>>> >>>>> > > -- > Saleem Abdulrasool > compnerd (at) compnerd (dot) org >-------------- next part -------------- An HTML attachment was scrubbed... URL: <http://lists.llvm.org/pipermail/llvm-dev/attachments/20190325/a03961c1/attachment-0001.html>
Yes, I don't think that supporting all options is entirely possible. But, there is certainly some allure to it, as it is a really powerful feature (which you can tell from the behaviour of link and `.drectve` section). The problem with all options is that ELF's model is not as simplistic as link's model, which really complicates things. As an example: `/WHOLEARCHIVE:` rather than `--whole-archive` ... `--no-whole-archive`. I'm also worried about its use in a malicious manner: `--no-pie` being embedded into the executable would disable ASLR, which really is why I think that we would want a subset of options. On Mon, Mar 25, 2019 at 8:47 AM Rui Ueyama <ruiu at google.com> wrote:> On Sun, Mar 24, 2019 at 9:51 PM Saleem Abdulrasool <compnerd at compnerd.org> > wrote: > >> Sorry for the late chiming in. >> >> Yes, swift does use autolinking, and I would like to use that on all the >> targets. The only target which does not support this functionality >> currently are ELF based. That said, I think that `#pragma comment(link, >> ...)` is insufficient for my needs. Building Foundation requires framework >> style linking as well. The original design that I had in mind was derived >> from ld64 and link. Personally, I still strongly favour link's behaviour >> of parsing "command line" options from the object files when they are >> loaded. There was strong opposition to that approach from Rui though. >> Would we want to have special pragmas for each "feature"? >> > > I think as you double-quoted the words "command line", I believe we are on > the same page, but what I opposed was to allow arbitrary command line > options in the .linker-options section. That is really a can of worms that > we don't want to open. I didn't oppose allowing a limited set of > command-line-ish directives. That said I'd prefer the proposal of this > thread over the .linker-options though. > > The ELF model doesn't have the simplistic model for processing the command >> line that PE/COFF does. Because ordering is relevant to the model, it >> would be ideal to process them inline, but, since lld already moves far >> enough away from the traditional Unix model, perhaps we can simplify it to >> append the command line directives to the end of the command line. >> >> The other case that is interesting to think about is the autolinking >> support in C++ (and clang) modules. >> >> On Thu, Mar 21, 2019 at 9:49 AM bd1976 llvm <bd1976llvm at gmail.com> wrote: >> >>> On Thu, Mar 21, 2019 at 12:06 AM Rui Ueyama <ruiu at google.com> wrote: >>> >>>> Perhaps there's no one clean way to solve this issue, because >>>> previously all libraries and object files are explicitly given to the >>>> linker via a command line and the order of files in the command line >>>> matters. That assumes human intervention to work correctly. Now, the >>>> autolinking feature will add libraries implicitly. Since it's implicit, >>>> there will be only one way how that works, so sometimes that works and >>>> sometimes doesn't. >>>> >>>> It feels to me that we should aim for making it work reasonably well >>>> for reasonable use cases. By reasonable use cases, I'm thinking of the >>>> following: >>>> >>>> 1. --static option may or may not be given (i.e. we should allow that >>>> feature for both static linking and dynamic linking.) >>>> 2. There are no competing defined symbols in a given set of libraries, >>>> or if they exist, the program owner doesn't care which is linked to their >>>> program. >>>> 3. There may be circular dependencies between libraries. >>>> >>>> I don't think the above assumption is too odd. If I have to implement >>>> the autolinking feature to GNU linker for the above scenario, I'd probably >>>> use the following scheme: >>>> >>>> 1. While reading object files, memorize libraries that are autolinked >>>> 2. After linking everything, create a list of files consisting of >>>> autolinked libraries AND libraries given via the command line >>>> 3. Visit each file in the list as if they were wrapped in >>>> --start-group and --end-group. >>>> >>>> I'd think the above scheme should work reasonably well. What do you >>>> think? >>>> >>> >>> Very nice. I agree with your definition of "reasonable" usecaes >>> (actually, as I have said before, I think that restricting autolinking to >>> this "reasonable" set is actually a feature - to avoid developers having >>> source code that only works with a particular linker). I also like the >>> proposal for a GNU implementation - I think this is enough to show that >>> GNU-like linkers could implement this. >>> >>> At this point I will try to prototype this up so that people have an >>> implementation to play with. >>> >>> I am keen to hear from Saleem (compnerd) on this, as he did the >>> original .linker-options work. >>> >>> >>>> >>>> On Tue, Mar 19, 2019 at 11:02 AM bd1976 llvm <bd1976llvm at gmail.com> >>>> wrote: >>>> >>>>> On Mon, Mar 18, 2019 at 8:02 PM Rui Ueyama <ruiu at google.com> wrote: >>>>> >>>>>> On Thu, Mar 14, 2019 at 1:05 PM bd1976 llvm via llvm-dev < >>>>>> llvm-dev at lists.llvm.org> wrote: >>>>>> >>>>>>> On Thu, Mar 14, 2019 at 6:27 PM Peter Collingbourne <peter at pcc.me.uk> >>>>>>> wrote: >>>>>>> >>>>>>>> >>>>>>>> >>>>>>>> On Thu, Mar 14, 2019 at 6:08 AM bd1976 llvm via llvm-dev < >>>>>>>> llvm-dev at lists.llvm.org> wrote: >>>>>>>> >>>>>>>>> At Sony we offer autolinking as a feature in our ELF toolchain. We >>>>>>>>> would like to see full support for this feature upstream as there is >>>>>>>>> anecdotal evidence that it would find use beyond Sony. >>>>>>>>> >>>>>>>>> In general autolinking (https://en.wikipedia.org/wiki/Auto-linking) >>>>>>>>> allows developers to specify inputs to the linker in their source code. >>>>>>>>> LLVM and Clang already have support for autolinking on ELF via embedding >>>>>>>>> strings, which specify linker behavior, into a .linker-options section in >>>>>>>>> relocatable object files, see: >>>>>>>>> >>>>>>>>> RFC - >>>>>>>>> http://lists.llvm.org/pipermail/llvm-dev/2018-January/120101.html >>>>>>>>> LLVM - >>>>>>>>> https://llvm.org/docs/Extensions.html#linker-options-section-linker-options, >>>>>>>>> https://reviews.llvm.org/D40849 >>>>>>>>> Clang - >>>>>>>>> https://clang.llvm.org/docs/LanguageExtensions.html#specifying-linker-options-on-elf-targets, >>>>>>>>> https://reviews.llvm.org/D42758 >>>>>>>>> >>>>>>>>> However, although support was added to Clang and LLVM, no support >>>>>>>>> has been implemented in LLD; and, I get the sense, from reading the >>>>>>>>> reviews, that there wasn't agreement on the implementation when the changes >>>>>>>>> landed. The original motivation seems to have been to remove the >>>>>>>>> "autolink-extract" mechanism used by Swift to workaround the lack of >>>>>>>>> autolinking support for ELF. However, looking at the Swift source code, >>>>>>>>> Swift still seems to be using the "autolink-extract" method. >>>>>>>>> >>>>>>>>> So my first question: Are there any users of the current >>>>>>>>> implementation for ELF? >>>>>>>>> >>>>>>>>> Assuming that no one is using the current code, I would like to >>>>>>>>> suggest a different mechanism for autolinking. >>>>>>>>> >>>>>>>>> For ELF we need limited autolinking support. Specifically, we only >>>>>>>>> need support for "comment lib" pragmas ( >>>>>>>>> https://docs.microsoft.com/en-us/cpp/preprocessor/comment-c-cpp?view=vs-2017) >>>>>>>>> in C/C++ e.g. #pragma comment(lib, "foo"). My suggestion that we keep the >>>>>>>>> implementation as lean as possible. >>>>>>>>> >>>>>>>>> Principles to guide the implementation: >>>>>>>>> - Developers should be able to easily understand autolinking >>>>>>>>> behavior. >>>>>>>>> - Developers should be able to override autolinking from the >>>>>>>>> linker command line. >>>>>>>>> - Inputs specified via pragmas should be handled in a general way >>>>>>>>> to allow the same source code to work in different environments. >>>>>>>>> >>>>>>>>> I would like to propose that we focus on autolinking exclusively >>>>>>>>> and that we divorce the implementation from the idea of "linker options" >>>>>>>>> which, by nature, would tie source code to the vagaries of particular >>>>>>>>> linkers. I don't see much value in supporting other linker operations so I >>>>>>>>> suggest that the binary representation be a mergable string section >>>>>>>>> (SHF_MERGE, SHF_STRINGS), called .autolink, with custom type >>>>>>>>> SHT_LLVM_AUTOLINK (0x6fff4c04), and SHF_EXCLUDE set (to avoid the contents >>>>>>>>> appearing in the output). The compiler can form this section by >>>>>>>>> concatenating the arguments of the "comment lib" pragmas in the order they >>>>>>>>> are encountered. Partial (-r, -Ur) links can be handled by concatenating >>>>>>>>> .autolink sections with the normal mergeable string section rules. The >>>>>>>>> current .linker-options can remain (or be removed); but, "comment lib" >>>>>>>>> pragmas for ELF should be lowered to .autolink not to .linker-options. This >>>>>>>>> makes sense as there is no linker option that "comment lib" pragmas map >>>>>>>>> directly to. As an example, #pragma comment(lib, "foo") would result in: >>>>>>>>> >>>>>>>>> .section ".autolink","eMS", at llvm_autolink,1 >>>>>>>>> .asciz "foo" >>>>>>>>> >>>>>>>>> For LTO, equivalent information to the contents of a the .autolink >>>>>>>>> section will be written to the IRSymtab so that it is available to the >>>>>>>>> linker for symbol resolution. >>>>>>>>> >>>>>>>>> The linker will process the .autolink strings in the following way: >>>>>>>>> >>>>>>>>> 1. Inputs from the .autolink sections of a relocatable object file >>>>>>>>> are added when the linker decides to include that file (which could itself >>>>>>>>> be in a library) in the link. Autolinked inputs behave as if they were >>>>>>>>> appended to the command line as a group after all other options. As a >>>>>>>>> consequence the set of autolinked libraries are searched last to resolve >>>>>>>>> symbols. >>>>>>>>> >>>>>>>> >>>>>>>> If we want this to be compatible with GNU linkers, doesn't the >>>>>>>> autolinked input need to appear at the point immediately after the object >>>>>>>> file appears in the link? I'm imagining the case where you have a >>>>>>>> statically linked libc as well as a libbar.a autolinked from a foo.o. The >>>>>>>> link command line would look like this: >>>>>>>> >>>>>>>> ld foo.o -lc >>>>>>>> >>>>>>>> Now foo.o autolinks against bar. The command line becomes: >>>>>>>> >>>>>>>> ld foo.o -lc -lbar >>>>>>>> >>>>>>> >>>>>>> Actually, I was thinking that on a GNU linker the command line would >>>>>>> become "ld foo.o -lc -( -lbar )-"; but, this doesn't affect your point. >>>>>>> >>>>>>> >>>>>>>> >>>>>>>> If libbar.a requires an additional object file from libc.a, it will >>>>>>>> not be added to the link. >>>>>>>> >>>>>>>> >>>>>>> As it stands all the dependencies of an autolinked library must >>>>>>> themselves be autolinked. I had imagined that this is a reasonable >>>>>>> limitation. If not we need another scheme. I try to think about some >>>>>>> motivating examples for this. >>>>>>> >>>>>>> >>>>>>>> 2. It is an error if a file cannot be found for a given string. >>>>>>>>> 3. Any command line options in effect at the end of the command >>>>>>>>> line parsing apply to autolinked inputs, e.g. --whole-archive. >>>>>>>>> 4. Duplicate autolinked inputs are ignored. >>>>>>>>> >>>>>>>> >>>>>>>> This seems like it would work in GNU linkers, as long as the >>>>>>>> autolinked file is added to the link immediately after the last mention, >>>>>>>> rather than the first. Otherwise a command line like: >>>>>>>> >>>>>>>> ld foo1.o foo2.o >>>>>>>> >>>>>>>> (where foo1.o and foo2.o both autolink bar) could end up looking >>>>>>>> like: >>>>>>>> >>>>>>>> ld foo1.o -lbar foo2.o >>>>>>>> >>>>>>>> and you will not link anything from libbar.a that only foo2.o >>>>>>>> requires. It may end up being simpler to not ignore duplicates. >>>>>>>> >>>>>>> >>>>>>> Correct; but, given that the proposal was to handle the libraries as >>>>>>> if they are appended to the link line after everything on the command line >>>>>>> then I think this will work. With deduplication (and the use of SHF_MERGE) >>>>>>> developers get no ordering guarantees. I claim that this is a feature! My >>>>>>> rationale is that the order in which libraries are linked affects different >>>>>>> linkers in different ways (e.g. LLD does not resolve symbols from archives >>>>>>> in a compatible manner with either the Microsoft linker or the GNU >>>>>>> linkers.), by not allowing the user to control the order I am essentially >>>>>>> saying that autolinking is not suitable for libraries that offer competing >>>>>>> copies of the same symbol. This ties into my argument that "comment lib" >>>>>>> pragmas should be handled in as "general" a way as possible. >>>>>>> >>>>>> >>>>>> Right. I think if you need a fine control over the link order, >>>>>> autolinking is not a feature you want to use. Or, in general, if your >>>>>> program is sensitive to a link order because its source object files have >>>>>> competing symbols of the same name, it's perhaps unnecessarily fragile. >>>>>> >>>>>> That being said, I think you need to address the issue that pcc >>>>>> pointed out. If you statically link a program `foo` with the following >>>>>> command line >>>>>> >>>>>> ld -o foo foo.o -lc >>>>>> >>>>>> , `foo.o` auto-imports libbar.a, and libbar.a depends on libc.a, can >>>>>> your proposed feature pull out object files needed for libbar.a? >>>>>> >>>>> >>>>> It won't work on GNU linkers. It will work with LLD as LLD has >>>>> MSVC-like archive handling. However, I would like to make sure that >>>>> whatever we come up with can be supported in the GNU toolchain. >>>>> >>>>> I had thought that it would be acceptable that all the dependencies of >>>>> an autolinked library must themselves be autolinked in order to work on GNU >>>>> style linkers. Having thought more, I don't like this limitation - >>>>> especially as it doesn't exist for Microsoft style linkers. One possible >>>>> resolution could be that GNU linkers might have to implement another >>>>> command line option e.g. --auto-dep=<file> to allow injection into the >>>>> group of autolinked libraries. >>>>> >>>>> i.e In pcc's example you would need to do: "ld foo.o >>>>> --auto-dep=libc.a" which would become "ld --start-group libbar.a libc.a >>>>> --end-group" with autolinking. >>>>> >>>>> I wanted to avoid the approach of inserting autolinked libraries after >>>>> the object that autolinks them. In LLD (and MSVC) it becomes hard to reason >>>>> about "where" the linker is in the command line and it would also mean that >>>>> we can't have the nice separation between parsing the command line and >>>>> doing the rest of the link that we currently have. Also, if you give people >>>>> a way to have a fine grained control over the link order with autolinking >>>>> you risk ending up with source code that will link on GNU style linkers but >>>>> not with LLD (assuming GNU ever implemented support for autolinking). >>>>> >>>>> Scenario: >>>>> >>>>> libbar.a(bar.o) - defines symbol bar >>>>> libfoo.a(foo.o) - defines foo and autolinks libbar.a >>>>> main.o - references foo >>>>> another.o - does not reference foo >>>>> No references to bar exist >>>>> >>>>> lld -lfoo another.o --whole-archive main.o with autolinking becomes >>>>> lld -lfoo another.o --whole-archive main.o -lbar result: bar.o gets added >>>>> to the link. >>>>> But, if a change is made so that another.o references bar then the >>>>> link line with autolinking becomes lld -lfoo another.o >>>>> -lbar --whole-archive main.o result: bar.o is not added to the link. >>>>> >>>>> Hopefully the above scenario demonstrates why I think that it becomes >>>>> too complicated to reason about the effects of autolinking with pcc's >>>>> proposed insertion scheme. >>>>> >>>>> >>>>> >>>>>> 5. The linker tries to add a library or relocatable object file from >>>>>>>>> each of the strings in a .autolink section by; first, handling the string >>>>>>>>> as if it was specified on the commandline; second, by looking for the >>>>>>>>> string in each of the library search paths in turn; third, by looking for a >>>>>>>>> lib<string>.a or lib<string>.so (depending on the current mode of the >>>>>>>>> linker) in each of the library search paths. >>>>>>>>> >>>>>>>> >>>>>>>> Is the second part necessary? "-l:foo" causes the linker to search >>>>>>>> for a file named "foo" in the library search path, so it seems that >>>>>>>> allowing the autolink string to look like ":foo" would satisfy this use >>>>>>>> case. >>>>>>>> >>>>>>> >>>>>>> >>>>>>> I worded the proposal to avoid mapping "comment lib" pragmas to >>>>>>> --library command line options. My reasons: >>>>>>> >>>>>>> 1. I find the requirement that the user put ':' in their lib strings >>>>>>> slightly awkward. It means that the source code is now coupled to a >>>>>>> GNU-style linker. So then this isn't merely an ELF linking proposal, it's a >>>>>>> proposal for ELF toolchains with GNU-like linkers (e.g. the arm linker >>>>>>> doesn't support the colon prefix >>>>>>> http://infocenter.arm.com/help/index.jsp?topic=/com.arm.doc.dui0474c/Cjahbdei.html >>>>>>> ). >>>>>>> >>>>>>> 2. The syntax is #pragma comment(lib, ...) not #pragma >>>>>>> linker-option(library, ...) i.e. the only thing this (frankly rather >>>>>>> bizarre) syntax definitely implies is that the argument is related to >>>>>>> libraries (and comments ¯\_(ツ)_/¯); it is a bit of a stretch to interpret >>>>>>> "comment lib" pragmas as mapping directly to "specifying an additional >>>>>>> --library command line option". >>>>>>> >>>>>>> AFAIK all linkers support two ways of specifying inputs; firstly, >>>>>>> directly on the command line; secondly, with an option with very similar >>>>>>> semantics to GNU's --library option. I choose a method of finding a input >>>>>>> files that encompasses both methods of specifying a library on the command >>>>>>> line. I think that this method is actually more intuitive than either the >>>>>>> method used by the linker script INPUT command or by --library. FWIW, I >>>>>>> looked into the history of the colon prefix. It was added in >>>>>>> https://www.sourceware.org/ml/binutils/2007-03/msg00421.html. >>>>>>> Unfortunately, the rationale given is that it was merely a port of a >>>>>>> vxworks linker extension. I couldn't trace the history any further than >>>>>>> that to find the actual design discussion. The linker script command INPUT >>>>>>> uses a different scheme and the command already had this search order 20 >>>>>>> years ago, which is the earliest version of the GNU linker I have history >>>>>>> for; again, the rationale is not available. >>>>>>> >>>>>>> >>>>>>>> 6. A new command line option --no-llvm-autolink will tell LLD to >>>>>>>>> ignore the .autolink sections. >>>>>>>>> >>>>>>>>> Rationale for the above points: >>>>>>>>> >>>>>>>>> 1. Adding the autolinked inputs last makes the process simple to >>>>>>>>> understand from a developers perspective. All linkers are able to implement >>>>>>>>> this scheme. >>>>>>>>> 2. Error-ing for libraries that are not found seems like better >>>>>>>>> behavior than failing the link during symbol resolution. >>>>>>>>> 3. It seems useful for the user to be able to apply command line >>>>>>>>> options which will affect all of the autolinked input files. There is a >>>>>>>>> potential problem of surprise for developers, who might not realize that >>>>>>>>> these options would apply to the "invisible" autolinked input files; >>>>>>>>> however, despite the potential for surprise, this is easy for developers to >>>>>>>>> reason about and gives developers the control that they may require. >>>>>>>>> 4. Unlike on the command line it is probably easy to include the >>>>>>>>> same input file twice via pragmas and might be a pain to fix; think of >>>>>>>>> Third-party libraries supplied as binaries. >>>>>>>>> 5. This algorithm takes into account all of the different ways >>>>>>>>> that ELF linkers find input files. The different search methods are tried >>>>>>>>> by the linker in most obvious to least obvious order. >>>>>>>>> 6. I considered adding finer grained control over which .autolink >>>>>>>>> inputs were ignored (e.g. MSVC has /nodefaultlib:<library>); however, I >>>>>>>>> concluded that this is not necessary: if finer control is required >>>>>>>>> developers can recreate the same effect autolinking would have had using >>>>>>>>> command line options. >>>>>>>>> >>>>>>>>> Thoughts? >>>>>>>>> >>>>>>>>> _______________________________________________ >>>>>>>>> LLVM Developers mailing list >>>>>>>>> llvm-dev at lists.llvm.org >>>>>>>>> https://lists.llvm.org/cgi-bin/mailman/listinfo/llvm-dev >>>>>>>>> >>>>>>>> >>>>>>>> >>>>>>>> -- >>>>>>>> -- >>>>>>>> Peter >>>>>>>> >>>>>>> _______________________________________________ >>>>>>> LLVM Developers mailing list >>>>>>> llvm-dev at lists.llvm.org >>>>>>> https://lists.llvm.org/cgi-bin/mailman/listinfo/llvm-dev >>>>>>> >>>>>> >> >> -- >> Saleem Abdulrasool >> compnerd (at) compnerd (dot) org >> >-- Saleem Abdulrasool compnerd (at) compnerd (dot) org -------------- next part -------------- An HTML attachment was scrubbed... URL: <http://lists.llvm.org/pipermail/llvm-dev/attachments/20190325/21720f86/attachment-0001.html>
Thanks Saleem/James. On Tue, Mar 26, 2019 at 5:15 AM Saleem Abdulrasool <compnerd at compnerd.org> wrote:> Yes, I don't think that supporting all options is entirely possible. But, > there is certainly some allure to it, as it is a really powerful feature > (which you can tell from the behaviour of link and `.drectve` section). > The problem with all options is that ELF's model is not as simplistic as > link's model, which really complicates things. As an example: > `/WHOLEARCHIVE:` rather than `--whole-archive` ... `--no-whole-archive`. > I'm also worried about its use in a malicious manner: `--no-pie` being > embedded into the executable would disable ASLR, which really is why I > think that we would want a subset of options. >I'm not opposed to a linker options scheme; but, I think that it is important to separate this from autolinking (as we have proposed here). Rationale is upthread. We can support both if required.> On Mon, Mar 25, 2019 at 8:47 AM Rui Ueyama <ruiu at google.com> wrote: > >> On Sun, Mar 24, 2019 at 9:51 PM Saleem Abdulrasool <compnerd at compnerd.org> >> wrote: >> >>> Sorry for the late chiming in. >>> >>> Yes, swift does use autolinking, and I would like to use that on all the >>> targets. The only target which does not support this functionality >>> currently are ELF based. That said, I think that `#pragma comment(link, >>> ...)` is insufficient for my needs. Building Foundation requires framework >>> style linking as well. The original design that I had in mind was derived >>> from ld64 and link. Personally, I still strongly favour link's behaviour >>> of parsing "command line" options from the object files when they are >>> loaded. There was strong opposition to that approach from Rui though. >>> Would we want to have special pragmas for each "feature"? >>> >>I was hoping that the autolinking proposal here was general enough that it could be made to work for you. What if we made the library search algorithm even more general than I initially proposed. Something like: - The linker tries to add a library or relocatable object file from each of the strings in a .autolink section by; first, handling the string as if it was specified on the commandline; second, by looking for the string in each of the library search paths in turn; third, by looking for a lib<string>.a or lib<string>.so (depending on the current mode of the linker) in each of the library search paths; forth, by looking for the string in each of the framework search paths in turn. So if you have #pragma comment(lib, "foo") in a source file, then on a linker that supports framework linking the linker would add to the link the first matching file trying: ./foo <first library search path>/foo <second library search path>/foo .. <nth library search path>/foo <first library search path>/libfoo.so <second library search path>/libfoo.so .. <nth library search path>/libfoo.so <first library search path>/libfoo.a <second library search path>/libfoo.a .. <nth library search path>/libfoo.a <first framework search path>/foo.framework/foo <second library search path>/foo.framework/foo .. <nth library search path>/foo.framework/foo What do you think? I think as you double-quoted the words "command line", I believe we are on>> the same page, but what I opposed was to allow arbitrary command line >> options in the .linker-options section. That is really a can of worms that >> we don't want to open. I didn't oppose allowing a limited set of >> command-line-ish directives. That said I'd prefer the proposal of this >> thread over the .linker-options though. >> >> The ELF model doesn't have the simplistic model for processing the >>> command line that PE/COFF does. Because ordering is relevant to the model, >>> it would be ideal to process them inline, but, since lld already moves far >>> enough away from the traditional Unix model, perhaps we can simplify it to >>> append the command line directives to the end of the command line. >>> >>> The other case that is interesting to think about is the autolinking >>> support in C++ (and clang) modules. >>> >>When I started this work I looked at the C++ 20 modules proposal: http://www.open-std.org/jtc1/sc22/wg21/docs/papers/2019/p1103r3.pdf. I didn't see anything about autolinking (or linking at all)! Can you explain more?>>> On Thu, Mar 21, 2019 at 9:49 AM bd1976 llvm <bd1976llvm at gmail.com> >>> wrote: >>> >>>> On Thu, Mar 21, 2019 at 12:06 AM Rui Ueyama <ruiu at google.com> wrote: >>>> >>>>> Perhaps there's no one clean way to solve this issue, because >>>>> previously all libraries and object files are explicitly given to the >>>>> linker via a command line and the order of files in the command line >>>>> matters. That assumes human intervention to work correctly. Now, the >>>>> autolinking feature will add libraries implicitly. Since it's implicit, >>>>> there will be only one way how that works, so sometimes that works and >>>>> sometimes doesn't. >>>>> >>>>> It feels to me that we should aim for making it work reasonably well >>>>> for reasonable use cases. By reasonable use cases, I'm thinking of the >>>>> following: >>>>> >>>>> 1. --static option may or may not be given (i.e. we should allow that >>>>> feature for both static linking and dynamic linking.) >>>>> 2. There are no competing defined symbols in a given set of >>>>> libraries, or if they exist, the program owner doesn't care which is linked >>>>> to their program. >>>>> 3. There may be circular dependencies between libraries. >>>>> >>>>> I don't think the above assumption is too odd. If I have to implement >>>>> the autolinking feature to GNU linker for the above scenario, I'd probably >>>>> use the following scheme: >>>>> >>>>> 1. While reading object files, memorize libraries that are autolinked >>>>> 2. After linking everything, create a list of files consisting of >>>>> autolinked libraries AND libraries given via the command line >>>>> 3. Visit each file in the list as if they were wrapped in >>>>> --start-group and --end-group. >>>>> >>>>> I'd think the above scheme should work reasonably well. What do you >>>>> think? >>>>> >>>> >>>> Very nice. I agree with your definition of "reasonable" usecaes >>>> (actually, as I have said before, I think that restricting autolinking to >>>> this "reasonable" set is actually a feature - to avoid developers having >>>> source code that only works with a particular linker). I also like the >>>> proposal for a GNU implementation - I think this is enough to show that >>>> GNU-like linkers could implement this. >>>> >>>> At this point I will try to prototype this up so that people have an >>>> implementation to play with. >>>> >>>> I am keen to hear from Saleem (compnerd) on this, as he did the >>>> original .linker-options work. >>>> >>>> >>>>> >>>>> On Tue, Mar 19, 2019 at 11:02 AM bd1976 llvm <bd1976llvm at gmail.com> >>>>> wrote: >>>>> >>>>>> On Mon, Mar 18, 2019 at 8:02 PM Rui Ueyama <ruiu at google.com> wrote: >>>>>> >>>>>>> On Thu, Mar 14, 2019 at 1:05 PM bd1976 llvm via llvm-dev < >>>>>>> llvm-dev at lists.llvm.org> wrote: >>>>>>> >>>>>>>> On Thu, Mar 14, 2019 at 6:27 PM Peter Collingbourne < >>>>>>>> peter at pcc.me.uk> wrote: >>>>>>>> >>>>>>>>> >>>>>>>>> >>>>>>>>> On Thu, Mar 14, 2019 at 6:08 AM bd1976 llvm via llvm-dev < >>>>>>>>> llvm-dev at lists.llvm.org> wrote: >>>>>>>>> >>>>>>>>>> At Sony we offer autolinking as a feature in our ELF toolchain. >>>>>>>>>> We would like to see full support for this feature upstream as there is >>>>>>>>>> anecdotal evidence that it would find use beyond Sony. >>>>>>>>>> >>>>>>>>>> In general autolinking ( >>>>>>>>>> https://en.wikipedia.org/wiki/Auto-linking) allows developers to >>>>>>>>>> specify inputs to the linker in their source code. LLVM and Clang already >>>>>>>>>> have support for autolinking on ELF via embedding strings, which specify >>>>>>>>>> linker behavior, into a .linker-options section in relocatable object >>>>>>>>>> files, see: >>>>>>>>>> >>>>>>>>>> RFC - >>>>>>>>>> http://lists.llvm.org/pipermail/llvm-dev/2018-January/120101.html >>>>>>>>>> LLVM - >>>>>>>>>> https://llvm.org/docs/Extensions.html#linker-options-section-linker-options, >>>>>>>>>> https://reviews.llvm.org/D40849 >>>>>>>>>> Clang - >>>>>>>>>> https://clang.llvm.org/docs/LanguageExtensions.html#specifying-linker-options-on-elf-targets, >>>>>>>>>> https://reviews.llvm.org/D42758 >>>>>>>>>> >>>>>>>>>> However, although support was added to Clang and LLVM, no support >>>>>>>>>> has been implemented in LLD; and, I get the sense, from reading the >>>>>>>>>> reviews, that there wasn't agreement on the implementation when the changes >>>>>>>>>> landed. The original motivation seems to have been to remove the >>>>>>>>>> "autolink-extract" mechanism used by Swift to workaround the lack of >>>>>>>>>> autolinking support for ELF. However, looking at the Swift source code, >>>>>>>>>> Swift still seems to be using the "autolink-extract" method. >>>>>>>>>> >>>>>>>>>> So my first question: Are there any users of the current >>>>>>>>>> implementation for ELF? >>>>>>>>>> >>>>>>>>>> Assuming that no one is using the current code, I would like to >>>>>>>>>> suggest a different mechanism for autolinking. >>>>>>>>>> >>>>>>>>>> For ELF we need limited autolinking support. Specifically, we >>>>>>>>>> only need support for "comment lib" pragmas ( >>>>>>>>>> https://docs.microsoft.com/en-us/cpp/preprocessor/comment-c-cpp?view=vs-2017) >>>>>>>>>> in C/C++ e.g. #pragma comment(lib, "foo"). My suggestion that we keep the >>>>>>>>>> implementation as lean as possible. >>>>>>>>>> >>>>>>>>>> Principles to guide the implementation: >>>>>>>>>> - Developers should be able to easily understand autolinking >>>>>>>>>> behavior. >>>>>>>>>> - Developers should be able to override autolinking from the >>>>>>>>>> linker command line. >>>>>>>>>> - Inputs specified via pragmas should be handled in a general way >>>>>>>>>> to allow the same source code to work in different environments. >>>>>>>>>> >>>>>>>>>> I would like to propose that we focus on autolinking exclusively >>>>>>>>>> and that we divorce the implementation from the idea of "linker options" >>>>>>>>>> which, by nature, would tie source code to the vagaries of particular >>>>>>>>>> linkers. I don't see much value in supporting other linker operations so I >>>>>>>>>> suggest that the binary representation be a mergable string section >>>>>>>>>> (SHF_MERGE, SHF_STRINGS), called .autolink, with custom type >>>>>>>>>> SHT_LLVM_AUTOLINK (0x6fff4c04), and SHF_EXCLUDE set (to avoid the contents >>>>>>>>>> appearing in the output). The compiler can form this section by >>>>>>>>>> concatenating the arguments of the "comment lib" pragmas in the order they >>>>>>>>>> are encountered. Partial (-r, -Ur) links can be handled by concatenating >>>>>>>>>> .autolink sections with the normal mergeable string section rules. The >>>>>>>>>> current .linker-options can remain (or be removed); but, "comment lib" >>>>>>>>>> pragmas for ELF should be lowered to .autolink not to .linker-options. This >>>>>>>>>> makes sense as there is no linker option that "comment lib" pragmas map >>>>>>>>>> directly to. As an example, #pragma comment(lib, "foo") would result in: >>>>>>>>>> >>>>>>>>>> .section ".autolink","eMS", at llvm_autolink,1 >>>>>>>>>> .asciz "foo" >>>>>>>>>> >>>>>>>>>> For LTO, equivalent information to the contents of a the >>>>>>>>>> .autolink section will be written to the IRSymtab so that it is available >>>>>>>>>> to the linker for symbol resolution. >>>>>>>>>> >>>>>>>>>> The linker will process the .autolink strings in the following >>>>>>>>>> way: >>>>>>>>>> >>>>>>>>>> 1. Inputs from the .autolink sections of a relocatable object >>>>>>>>>> file are added when the linker decides to include that file (which could >>>>>>>>>> itself be in a library) in the link. Autolinked inputs behave as if they >>>>>>>>>> were appended to the command line as a group after all other options. As a >>>>>>>>>> consequence the set of autolinked libraries are searched last to resolve >>>>>>>>>> symbols. >>>>>>>>>> >>>>>>>>> >>>>>>>>> If we want this to be compatible with GNU linkers, doesn't the >>>>>>>>> autolinked input need to appear at the point immediately after the object >>>>>>>>> file appears in the link? I'm imagining the case where you have a >>>>>>>>> statically linked libc as well as a libbar.a autolinked from a foo.o. The >>>>>>>>> link command line would look like this: >>>>>>>>> >>>>>>>>> ld foo.o -lc >>>>>>>>> >>>>>>>>> Now foo.o autolinks against bar. The command line becomes: >>>>>>>>> >>>>>>>>> ld foo.o -lc -lbar >>>>>>>>> >>>>>>>> >>>>>>>> Actually, I was thinking that on a GNU linker the command line >>>>>>>> would become "ld foo.o -lc -( -lbar )-"; but, this doesn't affect your >>>>>>>> point. >>>>>>>> >>>>>>>> >>>>>>>>> >>>>>>>>> If libbar.a requires an additional object file from libc.a, it >>>>>>>>> will not be added to the link. >>>>>>>>> >>>>>>>>> >>>>>>>> As it stands all the dependencies of an autolinked library must >>>>>>>> themselves be autolinked. I had imagined that this is a reasonable >>>>>>>> limitation. If not we need another scheme. I try to think about some >>>>>>>> motivating examples for this. >>>>>>>> >>>>>>>> >>>>>>>>> 2. It is an error if a file cannot be found for a given string. >>>>>>>>>> 3. Any command line options in effect at the end of the command >>>>>>>>>> line parsing apply to autolinked inputs, e.g. --whole-archive. >>>>>>>>>> 4. Duplicate autolinked inputs are ignored. >>>>>>>>>> >>>>>>>>> >>>>>>>>> This seems like it would work in GNU linkers, as long as the >>>>>>>>> autolinked file is added to the link immediately after the last mention, >>>>>>>>> rather than the first. Otherwise a command line like: >>>>>>>>> >>>>>>>>> ld foo1.o foo2.o >>>>>>>>> >>>>>>>>> (where foo1.o and foo2.o both autolink bar) could end up looking >>>>>>>>> like: >>>>>>>>> >>>>>>>>> ld foo1.o -lbar foo2.o >>>>>>>>> >>>>>>>>> and you will not link anything from libbar.a that only foo2.o >>>>>>>>> requires. It may end up being simpler to not ignore duplicates. >>>>>>>>> >>>>>>>> >>>>>>>> Correct; but, given that the proposal was to handle the libraries >>>>>>>> as if they are appended to the link line after everything on the command >>>>>>>> line then I think this will work. With deduplication (and the use of >>>>>>>> SHF_MERGE) developers get no ordering guarantees. I claim that this is a >>>>>>>> feature! My rationale is that the order in which libraries are linked >>>>>>>> affects different linkers in different ways (e.g. LLD does not resolve >>>>>>>> symbols from archives in a compatible manner with either the Microsoft >>>>>>>> linker or the GNU linkers.), by not allowing the user to control the order >>>>>>>> I am essentially saying that autolinking is not suitable for libraries that >>>>>>>> offer competing copies of the same symbol. This ties into my argument that >>>>>>>> "comment lib" pragmas should be handled in as "general" a way as possible. >>>>>>>> >>>>>>> >>>>>>> Right. I think if you need a fine control over the link order, >>>>>>> autolinking is not a feature you want to use. Or, in general, if your >>>>>>> program is sensitive to a link order because its source object files have >>>>>>> competing symbols of the same name, it's perhaps unnecessarily fragile. >>>>>>> >>>>>>> That being said, I think you need to address the issue that pcc >>>>>>> pointed out. If you statically link a program `foo` with the following >>>>>>> command line >>>>>>> >>>>>>> ld -o foo foo.o -lc >>>>>>> >>>>>>> , `foo.o` auto-imports libbar.a, and libbar.a depends on libc.a, can >>>>>>> your proposed feature pull out object files needed for libbar.a? >>>>>>> >>>>>> >>>>>> It won't work on GNU linkers. It will work with LLD as LLD has >>>>>> MSVC-like archive handling. However, I would like to make sure that >>>>>> whatever we come up with can be supported in the GNU toolchain. >>>>>> >>>>>> I had thought that it would be acceptable that all the dependencies >>>>>> of an autolinked library must themselves be autolinked in order to work on >>>>>> GNU style linkers. Having thought more, I don't like this limitation - >>>>>> especially as it doesn't exist for Microsoft style linkers. One possible >>>>>> resolution could be that GNU linkers might have to implement another >>>>>> command line option e.g. --auto-dep=<file> to allow injection into the >>>>>> group of autolinked libraries. >>>>>> >>>>>> i.e In pcc's example you would need to do: "ld foo.o >>>>>> --auto-dep=libc.a" which would become "ld --start-group libbar.a libc.a >>>>>> --end-group" with autolinking. >>>>>> >>>>>> I wanted to avoid the approach of inserting autolinked libraries >>>>>> after the object that autolinks them. In LLD (and MSVC) it becomes hard to >>>>>> reason about "where" the linker is in the command line and it would also >>>>>> mean that we can't have the nice separation between parsing the command >>>>>> line and doing the rest of the link that we currently have. Also, if you >>>>>> give people a way to have a fine grained control over the link order with >>>>>> autolinking you risk ending up with source code that will link on GNU style >>>>>> linkers but not with LLD (assuming GNU ever implemented support for >>>>>> autolinking). >>>>>> >>>>>> Scenario: >>>>>> >>>>>> libbar.a(bar.o) - defines symbol bar >>>>>> libfoo.a(foo.o) - defines foo and autolinks libbar.a >>>>>> main.o - references foo >>>>>> another.o - does not reference foo >>>>>> No references to bar exist >>>>>> >>>>>> lld -lfoo another.o --whole-archive main.o with autolinking becomes >>>>>> lld -lfoo another.o --whole-archive main.o -lbar result: bar.o gets added >>>>>> to the link. >>>>>> But, if a change is made so that another.o references bar then the >>>>>> link line with autolinking becomes lld -lfoo another.o >>>>>> -lbar --whole-archive main.o result: bar.o is not added to the link. >>>>>> >>>>>> Hopefully the above scenario demonstrates why I think that it becomes >>>>>> too complicated to reason about the effects of autolinking with pcc's >>>>>> proposed insertion scheme. >>>>>> >>>>>> >>>>>> >>>>>>> 5. The linker tries to add a library or relocatable object file from >>>>>>>>>> each of the strings in a .autolink section by; first, handling the string >>>>>>>>>> as if it was specified on the commandline; second, by looking for the >>>>>>>>>> string in each of the library search paths in turn; third, by looking for a >>>>>>>>>> lib<string>.a or lib<string>.so (depending on the current mode of the >>>>>>>>>> linker) in each of the library search paths. >>>>>>>>>> >>>>>>>>> >>>>>>>>> Is the second part necessary? "-l:foo" causes the linker to search >>>>>>>>> for a file named "foo" in the library search path, so it seems that >>>>>>>>> allowing the autolink string to look like ":foo" would satisfy this use >>>>>>>>> case. >>>>>>>>> >>>>>>>> >>>>>>>> >>>>>>>> I worded the proposal to avoid mapping "comment lib" pragmas to >>>>>>>> --library command line options. My reasons: >>>>>>>> >>>>>>>> 1. I find the requirement that the user put ':' in their lib >>>>>>>> strings slightly awkward. It means that the source code is now coupled to a >>>>>>>> GNU-style linker. So then this isn't merely an ELF linking proposal, it's a >>>>>>>> proposal for ELF toolchains with GNU-like linkers (e.g. the arm linker >>>>>>>> doesn't support the colon prefix >>>>>>>> http://infocenter.arm.com/help/index.jsp?topic=/com.arm.doc.dui0474c/Cjahbdei.html >>>>>>>> ). >>>>>>>> >>>>>>>> 2. The syntax is #pragma comment(lib, ...) not #pragma >>>>>>>> linker-option(library, ...) i.e. the only thing this (frankly rather >>>>>>>> bizarre) syntax definitely implies is that the argument is related to >>>>>>>> libraries (and comments ¯\_(ツ)_/¯); it is a bit of a stretch to interpret >>>>>>>> "comment lib" pragmas as mapping directly to "specifying an additional >>>>>>>> --library command line option". >>>>>>>> >>>>>>>> AFAIK all linkers support two ways of specifying inputs; firstly, >>>>>>>> directly on the command line; secondly, with an option with very similar >>>>>>>> semantics to GNU's --library option. I choose a method of finding a input >>>>>>>> files that encompasses both methods of specifying a library on the command >>>>>>>> line. I think that this method is actually more intuitive than either the >>>>>>>> method used by the linker script INPUT command or by --library. FWIW, I >>>>>>>> looked into the history of the colon prefix. It was added in >>>>>>>> https://www.sourceware.org/ml/binutils/2007-03/msg00421.html. >>>>>>>> Unfortunately, the rationale given is that it was merely a port of a >>>>>>>> vxworks linker extension. I couldn't trace the history any further than >>>>>>>> that to find the actual design discussion. The linker script command INPUT >>>>>>>> uses a different scheme and the command already had this search order 20 >>>>>>>> years ago, which is the earliest version of the GNU linker I have history >>>>>>>> for; again, the rationale is not available. >>>>>>>> >>>>>>>> >>>>>>>>> 6. A new command line option --no-llvm-autolink will tell LLD to >>>>>>>>>> ignore the .autolink sections. >>>>>>>>>> >>>>>>>>>> Rationale for the above points: >>>>>>>>>> >>>>>>>>>> 1. Adding the autolinked inputs last makes the process simple to >>>>>>>>>> understand from a developers perspective. All linkers are able to implement >>>>>>>>>> this scheme. >>>>>>>>>> 2. Error-ing for libraries that are not found seems like better >>>>>>>>>> behavior than failing the link during symbol resolution. >>>>>>>>>> 3. It seems useful for the user to be able to apply command line >>>>>>>>>> options which will affect all of the autolinked input files. There is a >>>>>>>>>> potential problem of surprise for developers, who might not realize that >>>>>>>>>> these options would apply to the "invisible" autolinked input files; >>>>>>>>>> however, despite the potential for surprise, this is easy for developers to >>>>>>>>>> reason about and gives developers the control that they may require. >>>>>>>>>> 4. Unlike on the command line it is probably easy to include the >>>>>>>>>> same input file twice via pragmas and might be a pain to fix; think of >>>>>>>>>> Third-party libraries supplied as binaries. >>>>>>>>>> 5. This algorithm takes into account all of the different ways >>>>>>>>>> that ELF linkers find input files. The different search methods are tried >>>>>>>>>> by the linker in most obvious to least obvious order. >>>>>>>>>> 6. I considered adding finer grained control over which .autolink >>>>>>>>>> inputs were ignored (e.g. MSVC has /nodefaultlib:<library>); however, I >>>>>>>>>> concluded that this is not necessary: if finer control is required >>>>>>>>>> developers can recreate the same effect autolinking would have had using >>>>>>>>>> command line options. >>>>>>>>>> >>>>>>>>>> Thoughts? >>>>>>>>>> >>>>>>>>>> _______________________________________________ >>>>>>>>>> LLVM Developers mailing list >>>>>>>>>> llvm-dev at lists.llvm.org >>>>>>>>>> https://lists.llvm.org/cgi-bin/mailman/listinfo/llvm-dev >>>>>>>>>> >>>>>>>>> >>>>>>>>> >>>>>>>>> -- >>>>>>>>> -- >>>>>>>>> Peter >>>>>>>>> >>>>>>>> _______________________________________________ >>>>>>>> LLVM Developers mailing list >>>>>>>> llvm-dev at lists.llvm.org >>>>>>>> https://lists.llvm.org/cgi-bin/mailman/listinfo/llvm-dev >>>>>>>> >>>>>>> >>> >>> -- >>> Saleem Abdulrasool >>> compnerd (at) compnerd (dot) org >>> >> > > -- > Saleem Abdulrasool > compnerd (at) compnerd (dot) org >-------------- next part -------------- An HTML attachment was scrubbed... URL: <http://lists.llvm.org/pipermail/llvm-dev/attachments/20190326/e043b1b1/attachment.html>