This file is indexed.

/usr/share/doc/ubuntu-packaging-guide-html-uk/_sources/libraries.txt is in ubuntu-packaging-guide-html-uk 0.3.8.

This file is owned by root:root, with mode 0o644.

The actual contents of the file can be viewed below.

  1
  2
  3
  4
  5
  6
  7
  8
  9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 20
 21
 22
 23
 24
 25
 26
 27
 28
 29
 30
 31
 32
 33
 34
 35
 36
 37
 38
 39
 40
 41
 42
 43
 44
 45
 46
 47
 48
 49
 50
 51
 52
 53
 54
 55
 56
 57
 58
 59
 60
 61
 62
 63
 64
 65
 66
 67
 68
 69
 70
 71
 72
 73
 74
 75
 76
 77
 78
 79
 80
 81
 82
 83
 84
 85
 86
 87
 88
 89
 90
 91
 92
 93
 94
 95
 96
 97
 98
 99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
================
Shared Libraries
================

Shared libraries are compiled code which is intended to be shared
among several different programs.  They are distributed as ``.so``
files in ``/usr/lib/``.  

A library exports symbols which are the compiled versions of
functions, classes and variables.  A library has a name called an
SONAME which includes a version number.  This SONAME version does not
necessarily match the public release version number.  A program gets
compiled against a given SONAME version of the library.  If any of the
symbols is removed or changes then the version number needs to be
changed which forces any packages using that library to be recompiled
against the new version.  Version numbers are usually set by upstream
and we follow them in our binary package names called an ABI number,
but sometimes upstreams do not use sensible version numbers and
packagers have to keep separate version numbers.

Libraries are usually distributed by upstream as standalone releases. Sometimes
they are distributed as part of a program.  In this case they can be included
in the binary package along with the program (this is called bundling) if you
do not expect any other programs to use the library, more often they should be
split out into separate binary packages.

The libraries themselves are put into a binary package named ``libfoo1`` where
``foo`` is the name of the library and ``1`` is the version from the SONAME. 
Development files from the package, such as header files, needed to compile
programs against the library are put into a package called ``libfoo-dev``.


An Example
-----------

We will use libnova as an example::

    $ bzr branch ubuntu:trusty/libnova
    $ sudo apt-get install libnova-dev

To find the SONAME of the library run::

    $ readelf -a /usr/lib/libnova-0.12.so.2 | grep SONAME

The SONAME is ``libnova-0.12.so.2``, which matches the file name (usually the
case but not always). Here upstream has put the upstream version number as part
of the SONAME and given it an ABI version of ``2``.  Library package names
should follow the SONAME of the library they contain. The library binary package
is called ``libnova-0.12-2`` where ``libnova-0.12`` is the name of the library
and ``2`` is our ABI number.

If upstream makes incompatible changes to their library they will have to
reversion their SONAME and we will have to rename our library.  Any other
packages using our library package will need to recompiled against the new
version, this is called a transition and can take some effort. Hopefully our ABI
number will continue to match upstream's SONAME but sometimes they introduce
incompatibilities without changing their version number and we will need to
change ours. 

Looking in debian/libnova-0.12-2.install we see it includes two files::

    usr/lib/libnova-0.12.so.2
    usr/lib/libnova-0.12.so.2.0.0

The last one is the actual library, complete with minor and point version
number.  The first one is a symlink which points to the actual library.  The
symlink is what programs using the library will look for, the running
programs do not care about the minor version number.

``libnova-dev.install`` includes all the files needed to compile a program
with this library.  Header files, a config binary, the ``.la`` libtool file and
``libnova.so`` which is another symlink pointing at the library, programs
compiling against the library do not care about the major version number
(although the binary they compile into will).

``.la`` libtool files are needed on some non-Linux systems with poor library
support but usually cause more problems than they solve on Debian systems.  It
is a current `Debian goal to remove .la files <LAFileRemoval_>`_ and we should help with this.


Static Libraries
----------------

The -dev package also ships ``usr/lib/libnova.a``.  This is a static library,
an alternative to the shared library.  Any program compiled against the
static library will include the code directory into itself.  This gets round
worrying about binary compatibility of the library.  However it also means that
any bugs, including security issues, will not be updated along with the library
until the program is recompiled.  For this reason programs using static
libraries are discouraged.


Symbol Files
------------

When a package builds against a library the ``shlibs`` mechanism will add a
package dependency on that library.  This is why most programs will have
``Depends: ${shlibs:Depends}`` in ``debian/control``.  That gets replaced with
the library dependencies at build time.  However shlibs can only make it depend
on the major ABI version number, ``2`` in our libnova example, so if new symbols
get added in libnova 2.1 a program using these symbols could still be
installed against libnova ABI 2.0 which would then crash.

To make the library dependencies more precise we keep ``.symbols`` files that
list all the symbols in a library and the version they appeared in.

libnova has no symbols file so we can create one.  Start by compiling the
package::

    $ bzr builddeb -- -nc

The ``-nc`` will cause it to finish at the end of compilation without removing
the built files.  Change to the build and run ``dpkg-gensymbols`` for the
library package::

    $ cd ../build-area/libnova-0.12.2/
    $ dpkg-gensymbols -plibnova-0.12-2 > symbols.diff

This makes a diff file which you can self apply::

    $ patch -p0 < symbols.diff

Which will create a file named similar to ``dpkg-gensymbolsnY_WWI`` that lists
all the symbols.  It also lists the current package version.  We can remove the
packaging version from that listed in the symbols file because new symbols are
not generally added by new packaging versions, but by the upstream developers::

    $ sed -i s,-0ubuntu2,, dpkg-gensymbolsnY_WWI

Now move the file into its location, commit and do a test build::

    $ mv dpkg-gensymbolsnY_WWI ../../libnova/debian/libnova-0.12-2.symbols
    $ cd ../../libnova
    $ bzr add debian/libnova-0.12-2.symbols
    $ bzr commit -m "add symbols file"
    $ bzr builddeb

If it successfully compiles the symbols file is correct.  With the next
upstream version of libnova you would run dpkg-gensymbols again and it will
give a diff to update the symbols file.

C++ Library Symbols Files
-------------------------

C++ has even more exacting standards of binary compatibility than C.  The
Debian Qt/KDE Team maintain some scripts to handle this, see their `Working with
symbols files <SymbolsFiles_>`_ page for how to use them.

Further Reading
---------------

Junichi Uekawa's `Debian Library Packaging Guide <LibraryGuide_>`_ goes into this topic in more
detail.

.. _SymbolsFiles: http://pkg-kde.alioth.debian.org/symbolfiles.html
.. _LAFileRemoval: https://wiki.debian.org/ReleaseGoals/LAFileRemoval
.. _LibraryGuide: http://www.netfort.gr.jp/~dancer/column/libpkg-guide/libpkg-guide.html