rtld enhancement to add osversion, version 2
Konstantin Belousov
kostikbel at gmail.com
Mon Mar 19 21:03:59 UTC 2012
On Mon, Mar 19, 2012 at 04:59:34PM -0400, John Baldwin wrote:
> On Tuesday, March 13, 2012 1:29:17 pm Doug Ambrisko wrote:
> > This is round 2 of my rtld enhancements. The primary goal it to have
> > rtld look in different places for libraries depending on the legacy
> > binary that we want to run. This is especially a problem with binaries
> > linked to libraries from ports since the version of a library in
> > /usr/local/lib is the same whether it was FreeBSD 6, 7, 8 etc. until the
> > library itself changes version due to an interface change. At work we
> > need to run 3rd party binaries on our box of which we don't have source.
> > Having a FreeBSD 6 binaries load /usr/local/lib/libiconv.so.3 that was
> > built for FreeBSD 9 is not good. It is worse when libiconv.so.3 is
> > linked to libc.so.7 when the FreeBSD 6 binary needs libc.so.6.
> >
> > I solved this by having rtld extract the OSVERSION from the binary
> > and then use that to determine what to do. In my prior version,
> > I inserted that into the library directory. That meant to pull
> > in libc it would look at:
> > /lib/603000/libc.so.6
> > /lib/6/libc.so.6
> > /lib/libc.so.6
> > to find libc.so.6. This meant a lot more look ups. Also I found it
> > had a problem in that if we had an ambiguous name say
> > /usr/local/lib/libcrypto.so.5 on the FreeBSD 6 machine and
> /lib/libcrypto.so.5
> > on the new FreeBSD 8 system, just doing the search logic would get
> > a hit on /lib/libcrypto.so.5 before it got to /usr/local/lib/libcrypto.so.5.
> > So then it would mean we'd have to put all FreeBSD lib's in /lib/6 to
> > beat the search path. This wasn't a good solution. To solve the
> > performance and path issue, I now follow the 32 bit hints file name
> > model. Now it does a lookup of the hints file based on the osversion
> > and major. So now it looks for the hints file as:
> > /var/run/ld-elf-603000.so.hints
> > /var/run/ld-elf-6.so.hints
> > /var/run/ld-elf.so.hints
> > This is faster and has more unique paths for FreeBSD 6 libraries since
> > the FreeBSD 6 search paths would be in the hints file. I modified
> > ldconfig to accept an "-os=<version>" option to create this hints file.
> > I tweaked /etc/rc* to make this easy to setup like this:
> > ldconfig_os_versions="6"
> > ldconfig_6_path="/usr/local/lib/compat/6"
>
> I think this is a definite improvement from before, thanks!
>
> > This doesn't solve the LD_LIBRARY_PATH or LD_PRELOAD. Solving that
> > I still insert and iterate over the osverion, major and none into the
> > path to find the library. The reason for this is that a FreeBSD 8
> > binary could exec a FreeBSD 6 binary that execs a FreeBSD 7 binary.
> > If for the FreeBSD 6 binary we needed to set a custom LD_LIBRARY_PATH
> > and the FreeBSD 7 binary find a library via the FreeBSD 6 search path
> > then the FreeBSD 7 binary would die. By adding in the osversion search
> > path I can put the FreeBSD 6 library into the search path + the directory
> > 6. Then only FreeBSD 6 binaries can find it. An example:
> > LD_LIBRARY_PATH=/usr/custom_software/lib
> > /usr/custom_software/lib/6/libfoo.so.6
> > then only the FreeBSD 6 binary could load it. Since the searches
> > would be for the FreeBSD 6 binary:
> > /usr/custom_software/lib/603000/libfoo.so.6
> > /usr/custom_software/lib/6/libfoo.so.6
> > /usr/custom_software/lib/libfoo.so.6
> > and for FreeBSD 7 binary:
> > /usr/custom_software/lib/702000/libfoo.so.6
> > /usr/custom_software/lib/7/libfoo.so.6
> > /usr/custom_software/lib/libfoo.so.6
> > Only the FreeBSD 6 binary would load /usr/custom_software/lib/6/libfoo.so.6.
> > I do the same search with LD_PRELOAD.
>
> Hmm, I'm still not quite fan of this. Perhaps you could add an extension to
> ldconfig and the hints file to handle this case? That is, a way to store
> path mappings so you could do something like:
>
> ldconfig -os=6 -p /usr/local/lib /usr/local/lib/6
>
> Or maybe you could make it an extension of how 'm' worked, so you could make
> directories accept an optional set of colon-separated paths that they serve
> as aliases for:
>
> ldconfig -os=6 -m /usr/local/lib/6:/usr/local/lib:/usr/lib
>
> (That would even fit into your existing rc.d script changes I believe). Then
> rtld would keep this internal directory mapping and be able to map the
> '/usr/local/lib' and '/usr/lib' directories in LD_PRELOAD and LD_LIBRARY_PATH
> to /usr/local/lib/6. The advantage of this is the same as with your previous
> change that all the mappings are configurable and not hard-coded into rtld
> itself.
>
> > Final, is that prior binaries built on FreeBSD i386 but run on FreeBSD amd64
> > that set LD_* environemnt variables would fail on FreeBSD amd64 as is
> > since it didn't set the equivalent LD32_*. To address this for COMPAT_32BIT
> > I have rtld look for LD32_* first and then check for the LD_* second.
> > This way legacy applications work fine.
>
> Hmm, so Yahoo! had some patches to handle this as well. I think Yahoo's
> patches were different though. They actually patched the 32-bit libc to
> capture attempts to get or set LD_* and convert them to actually get/set
> LD32_* instead. I'm not sure which approach is best, but it might be worth
> asking Peter why Yahoo! did it that way and if there were reasons for that
> approach vs. doing it in rtld. I think the primary reason was so that you
> could set LD_LIBRARY_PATH or LD_PRELOAD to reference 64-bit libraries and
> not have it break 32-bit apps, but if a 32-bit app tried to set a variable
> before launching another app it would still DTRT.
>
> I do think this is definitely a problem worth solving.
Just a trivial note: r232831 added note parsing and stores osrel in
the Obj_Entry.
-------------- next part --------------
A non-text attachment was scrubbed...
Name: not available
Type: application/pgp-signature
Size: 196 bytes
Desc: not available
Url : http://lists.freebsd.org/pipermail/freebsd-arch/attachments/20120319/4e48a4a3/attachment.pgp
More information about the freebsd-arch
mailing list