2009-10-27 04:06:31 +08:00
|
|
|
<?xml version="1.0" encoding="UTF-8"?>
|
|
|
|
<!DOCTYPE book PUBLIC "-//OASIS//DTD DocBook XML V4.1.2//EN"
|
|
|
|
"http://www.oasis-open.org/docbook/xml/4.1.2/docbookx.dtd" []>
|
|
|
|
|
|
|
|
<book id="drmDevelopersGuide">
|
|
|
|
<bookinfo>
|
|
|
|
<title>Linux DRM Developer's Guide</title>
|
|
|
|
|
|
|
|
<copyright>
|
|
|
|
<year>2008-2009</year>
|
|
|
|
<holder>
|
|
|
|
Intel Corporation (Jesse Barnes <jesse.barnes@intel.com>)
|
|
|
|
</holder>
|
|
|
|
</copyright>
|
|
|
|
|
|
|
|
<legalnotice>
|
|
|
|
<para>
|
|
|
|
The contents of this file may be used under the terms of the GNU
|
|
|
|
General Public License version 2 (the "GPL") as distributed in
|
|
|
|
the kernel source COPYING file.
|
|
|
|
</para>
|
|
|
|
</legalnotice>
|
|
|
|
</bookinfo>
|
|
|
|
|
|
|
|
<toc></toc>
|
|
|
|
|
|
|
|
<!-- Introduction -->
|
|
|
|
|
|
|
|
<chapter id="drmIntroduction">
|
|
|
|
<title>Introduction</title>
|
|
|
|
<para>
|
|
|
|
The Linux DRM layer contains code intended to support the needs
|
|
|
|
of complex graphics devices, usually containing programmable
|
|
|
|
pipelines well suited to 3D graphics acceleration. Graphics
|
2011-08-26 01:21:31 +08:00
|
|
|
drivers in the kernel may make use of DRM functions to make
|
2009-10-27 04:06:31 +08:00
|
|
|
tasks like memory management, interrupt handling and DMA easier,
|
|
|
|
and provide a uniform interface to applications.
|
|
|
|
</para>
|
|
|
|
<para>
|
|
|
|
A note on versions: this guide covers features found in the DRM
|
|
|
|
tree, including the TTM memory manager, output configuration and
|
|
|
|
mode setting, and the new vblank internals, in addition to all
|
|
|
|
the regular features found in current kernels.
|
|
|
|
</para>
|
|
|
|
<para>
|
|
|
|
[Insert diagram of typical DRM stack here]
|
|
|
|
</para>
|
|
|
|
</chapter>
|
|
|
|
|
|
|
|
<!-- Internals -->
|
|
|
|
|
|
|
|
<chapter id="drmInternals">
|
|
|
|
<title>DRM Internals</title>
|
|
|
|
<para>
|
|
|
|
This chapter documents DRM internals relevant to driver authors
|
|
|
|
and developers working to add support for the latest features to
|
|
|
|
existing drivers.
|
|
|
|
</para>
|
|
|
|
<para>
|
2011-08-26 01:18:08 +08:00
|
|
|
First, we go over some typical driver initialization
|
2009-10-27 04:06:31 +08:00
|
|
|
requirements, like setting up command buffers, creating an
|
|
|
|
initial output configuration, and initializing core services.
|
2011-08-26 01:18:08 +08:00
|
|
|
Subsequent sections cover core internals in more detail,
|
2009-10-27 04:06:31 +08:00
|
|
|
providing implementation notes and examples.
|
|
|
|
</para>
|
|
|
|
<para>
|
|
|
|
The DRM layer provides several services to graphics drivers,
|
|
|
|
many of them driven by the application interfaces it provides
|
|
|
|
through libdrm, the library that wraps most of the DRM ioctls.
|
|
|
|
These include vblank event handling, memory
|
|
|
|
management, output management, framebuffer management, command
|
|
|
|
submission & fencing, suspend/resume support, and DMA
|
|
|
|
services.
|
|
|
|
</para>
|
|
|
|
<para>
|
2011-02-14 17:58:23 +08:00
|
|
|
The core of every DRM driver is struct drm_driver. Drivers
|
2011-08-26 01:18:08 +08:00
|
|
|
typically statically initialize a drm_driver structure,
|
2009-10-27 04:06:31 +08:00
|
|
|
then pass it to drm_init() at load time.
|
|
|
|
</para>
|
|
|
|
|
|
|
|
<!-- Internals: driver init -->
|
|
|
|
|
|
|
|
<sect1>
|
|
|
|
<title>Driver initialization</title>
|
|
|
|
<para>
|
|
|
|
Before calling the DRM initialization routines, the driver must
|
2011-02-14 17:58:23 +08:00
|
|
|
first create and fill out a struct drm_driver structure.
|
2009-10-27 04:06:31 +08:00
|
|
|
</para>
|
|
|
|
<programlisting>
|
|
|
|
static struct drm_driver driver = {
|
2011-08-26 01:55:54 +08:00
|
|
|
/* Don't use MTRRs here; the Xserver or userspace app should
|
|
|
|
* deal with them for Intel hardware.
|
2009-10-27 04:06:31 +08:00
|
|
|
*/
|
|
|
|
.driver_features =
|
|
|
|
DRIVER_USE_AGP | DRIVER_REQUIRE_AGP |
|
|
|
|
DRIVER_HAVE_IRQ | DRIVER_IRQ_SHARED | DRIVER_MODESET,
|
|
|
|
.load = i915_driver_load,
|
|
|
|
.unload = i915_driver_unload,
|
|
|
|
.firstopen = i915_driver_firstopen,
|
|
|
|
.lastclose = i915_driver_lastclose,
|
|
|
|
.preclose = i915_driver_preclose,
|
|
|
|
.save = i915_save,
|
|
|
|
.restore = i915_restore,
|
|
|
|
.device_is_agp = i915_driver_device_is_agp,
|
|
|
|
.get_vblank_counter = i915_get_vblank_counter,
|
|
|
|
.enable_vblank = i915_enable_vblank,
|
|
|
|
.disable_vblank = i915_disable_vblank,
|
|
|
|
.irq_preinstall = i915_driver_irq_preinstall,
|
|
|
|
.irq_postinstall = i915_driver_irq_postinstall,
|
|
|
|
.irq_uninstall = i915_driver_irq_uninstall,
|
|
|
|
.irq_handler = i915_driver_irq_handler,
|
|
|
|
.reclaim_buffers = drm_core_reclaim_buffers,
|
|
|
|
.get_map_ofs = drm_core_get_map_ofs,
|
|
|
|
.get_reg_ofs = drm_core_get_reg_ofs,
|
|
|
|
.fb_probe = intelfb_probe,
|
|
|
|
.fb_remove = intelfb_remove,
|
|
|
|
.fb_resize = intelfb_resize,
|
|
|
|
.master_create = i915_master_create,
|
|
|
|
.master_destroy = i915_master_destroy,
|
|
|
|
#if defined(CONFIG_DEBUG_FS)
|
|
|
|
.debugfs_init = i915_debugfs_init,
|
|
|
|
.debugfs_cleanup = i915_debugfs_cleanup,
|
|
|
|
#endif
|
|
|
|
.gem_init_object = i915_gem_init_object,
|
|
|
|
.gem_free_object = i915_gem_free_object,
|
|
|
|
.gem_vm_ops = &i915_gem_vm_ops,
|
|
|
|
.ioctls = i915_ioctls,
|
|
|
|
.fops = {
|
|
|
|
.owner = THIS_MODULE,
|
|
|
|
.open = drm_open,
|
|
|
|
.release = drm_release,
|
|
|
|
.ioctl = drm_ioctl,
|
|
|
|
.mmap = drm_mmap,
|
|
|
|
.poll = drm_poll,
|
|
|
|
.fasync = drm_fasync,
|
|
|
|
#ifdef CONFIG_COMPAT
|
|
|
|
.compat_ioctl = i915_compat_ioctl,
|
|
|
|
#endif
|
2010-07-07 00:54:47 +08:00
|
|
|
.llseek = noop_llseek,
|
2009-10-27 04:06:31 +08:00
|
|
|
},
|
|
|
|
.pci_driver = {
|
|
|
|
.name = DRIVER_NAME,
|
|
|
|
.id_table = pciidlist,
|
|
|
|
.probe = probe,
|
|
|
|
.remove = __devexit_p(drm_cleanup_pci),
|
|
|
|
},
|
|
|
|
.name = DRIVER_NAME,
|
|
|
|
.desc = DRIVER_DESC,
|
|
|
|
.date = DRIVER_DATE,
|
|
|
|
.major = DRIVER_MAJOR,
|
|
|
|
.minor = DRIVER_MINOR,
|
|
|
|
.patchlevel = DRIVER_PATCHLEVEL,
|
|
|
|
};
|
|
|
|
</programlisting>
|
|
|
|
<para>
|
|
|
|
In the example above, taken from the i915 DRM driver, the driver
|
2011-08-26 02:10:12 +08:00
|
|
|
sets several flags indicating what core features it supports;
|
|
|
|
we go over the individual callbacks in later sections. Since
|
2009-10-27 04:06:31 +08:00
|
|
|
flags indicate which features your driver supports to the DRM
|
|
|
|
core, you need to set most of them prior to calling drm_init(). Some,
|
|
|
|
like DRIVER_MODESET can be set later based on user supplied parameters,
|
|
|
|
but that's the exception rather than the rule.
|
|
|
|
</para>
|
|
|
|
<variablelist>
|
|
|
|
<title>Driver flags</title>
|
|
|
|
<varlistentry>
|
|
|
|
<term>DRIVER_USE_AGP</term>
|
|
|
|
<listitem><para>
|
|
|
|
Driver uses AGP interface
|
|
|
|
</para></listitem>
|
|
|
|
</varlistentry>
|
|
|
|
<varlistentry>
|
|
|
|
<term>DRIVER_REQUIRE_AGP</term>
|
|
|
|
<listitem><para>
|
|
|
|
Driver needs AGP interface to function.
|
|
|
|
</para></listitem>
|
|
|
|
</varlistentry>
|
|
|
|
<varlistentry>
|
|
|
|
<term>DRIVER_USE_MTRR</term>
|
|
|
|
<listitem>
|
|
|
|
<para>
|
|
|
|
Driver uses MTRR interface for mapping memory. Deprecated.
|
|
|
|
</para>
|
|
|
|
</listitem>
|
|
|
|
</varlistentry>
|
|
|
|
<varlistentry>
|
|
|
|
<term>DRIVER_PCI_DMA</term>
|
|
|
|
<listitem><para>
|
|
|
|
Driver is capable of PCI DMA. Deprecated.
|
|
|
|
</para></listitem>
|
|
|
|
</varlistentry>
|
|
|
|
<varlistentry>
|
|
|
|
<term>DRIVER_SG</term>
|
|
|
|
<listitem><para>
|
|
|
|
Driver can perform scatter/gather DMA. Deprecated.
|
|
|
|
</para></listitem>
|
|
|
|
</varlistentry>
|
|
|
|
<varlistentry>
|
|
|
|
<term>DRIVER_HAVE_DMA</term>
|
|
|
|
<listitem><para>Driver supports DMA. Deprecated.</para></listitem>
|
|
|
|
</varlistentry>
|
|
|
|
<varlistentry>
|
|
|
|
<term>DRIVER_HAVE_IRQ</term><term>DRIVER_IRQ_SHARED</term>
|
|
|
|
<listitem>
|
|
|
|
<para>
|
2011-08-26 02:20:54 +08:00
|
|
|
DRIVER_HAVE_IRQ indicates whether the driver has an IRQ
|
2011-08-26 02:21:29 +08:00
|
|
|
handler. DRIVER_IRQ_SHARED indicates whether the device &
|
2009-10-27 04:06:31 +08:00
|
|
|
handler support shared IRQs (note that this is required of
|
|
|
|
PCI drivers).
|
|
|
|
</para>
|
|
|
|
</listitem>
|
|
|
|
</varlistentry>
|
|
|
|
<varlistentry>
|
|
|
|
<term>DRIVER_DMA_QUEUE</term>
|
|
|
|
<listitem>
|
|
|
|
<para>
|
2011-08-26 02:23:21 +08:00
|
|
|
Should be set if the driver queues DMA requests and completes them
|
|
|
|
asynchronously. Deprecated.
|
2009-10-27 04:06:31 +08:00
|
|
|
</para>
|
|
|
|
</listitem>
|
|
|
|
</varlistentry>
|
|
|
|
<varlistentry>
|
|
|
|
<term>DRIVER_FB_DMA</term>
|
|
|
|
<listitem>
|
|
|
|
<para>
|
|
|
|
Driver supports DMA to/from the framebuffer. Deprecated.
|
|
|
|
</para>
|
|
|
|
</listitem>
|
|
|
|
</varlistentry>
|
|
|
|
<varlistentry>
|
|
|
|
<term>DRIVER_MODESET</term>
|
|
|
|
<listitem>
|
|
|
|
<para>
|
|
|
|
Driver supports mode setting interfaces.
|
|
|
|
</para>
|
|
|
|
</listitem>
|
|
|
|
</varlistentry>
|
|
|
|
</variablelist>
|
|
|
|
<para>
|
|
|
|
In this specific case, the driver requires AGP and supports
|
2011-08-26 01:18:08 +08:00
|
|
|
IRQs. DMA, as discussed later, is handled by device specific ioctls
|
2009-10-27 04:06:31 +08:00
|
|
|
in this case. It also supports the kernel mode setting APIs, though
|
|
|
|
unlike in the actual i915 driver source, this example unconditionally
|
|
|
|
exports KMS capability.
|
|
|
|
</para>
|
|
|
|
</sect1>
|
|
|
|
|
|
|
|
<!-- Internals: driver load -->
|
|
|
|
|
|
|
|
<sect1>
|
|
|
|
<title>Driver load</title>
|
|
|
|
<para>
|
|
|
|
In the previous section, we saw what a typical drm_driver
|
|
|
|
structure might look like. One of the more important fields in
|
|
|
|
the structure is the hook for the load function.
|
|
|
|
</para>
|
|
|
|
<programlisting>
|
|
|
|
static struct drm_driver driver = {
|
|
|
|
...
|
|
|
|
.load = i915_driver_load,
|
|
|
|
...
|
|
|
|
};
|
|
|
|
</programlisting>
|
|
|
|
<para>
|
|
|
|
The load function has many responsibilities: allocating a driver
|
|
|
|
private structure, specifying supported performance counters,
|
|
|
|
configuring the device (e.g. mapping registers & command
|
|
|
|
buffers), initializing the memory manager, and setting up the
|
|
|
|
initial output configuration.
|
|
|
|
</para>
|
|
|
|
<para>
|
2011-08-26 02:37:05 +08:00
|
|
|
If compatibility is a concern (e.g. with drivers converted over
|
|
|
|
to the new interfaces from the old ones), care must be taken to
|
|
|
|
prevent device initialization and control that is incompatible with
|
|
|
|
currently active userspace drivers. For instance, if user
|
2009-10-27 04:06:31 +08:00
|
|
|
level mode setting drivers are in use, it would be problematic
|
|
|
|
to perform output discovery & configuration at load time.
|
2011-08-26 02:31:42 +08:00
|
|
|
Likewise, if user-level drivers unaware of memory management are
|
2009-10-27 04:06:31 +08:00
|
|
|
in use, memory management and command buffer setup may need to
|
|
|
|
be omitted. These requirements are driver specific, and care
|
|
|
|
needs to be taken to keep both old and new applications and
|
|
|
|
libraries working. The i915 driver supports the "modeset"
|
|
|
|
module parameter to control whether advanced features are
|
2011-08-26 02:37:05 +08:00
|
|
|
enabled at load time or in legacy fashion.
|
2009-10-27 04:06:31 +08:00
|
|
|
</para>
|
|
|
|
|
|
|
|
<sect2>
|
|
|
|
<title>Driver private & performance counters</title>
|
|
|
|
<para>
|
|
|
|
The driver private hangs off the main drm_device structure and
|
|
|
|
can be used for tracking various device specific bits of
|
|
|
|
information, like register offsets, command buffer status,
|
|
|
|
register state for suspend/resume, etc. At load time, a
|
2011-08-26 01:21:31 +08:00
|
|
|
driver may simply allocate one and set drm_device.dev_priv
|
2011-08-26 02:40:55 +08:00
|
|
|
appropriately; it should be freed and drm_device.dev_priv set
|
|
|
|
to NULL when the driver is unloaded.
|
2009-10-27 04:06:31 +08:00
|
|
|
</para>
|
|
|
|
<para>
|
2011-08-26 01:21:31 +08:00
|
|
|
The DRM supports several counters which may be used for rough
|
2009-10-27 04:06:31 +08:00
|
|
|
performance characterization. Note that the DRM stat counter
|
|
|
|
system is not often used by applications, and supporting
|
|
|
|
additional counters is completely optional.
|
|
|
|
</para>
|
|
|
|
<para>
|
|
|
|
These interfaces are deprecated and should not be used. If performance
|
|
|
|
monitoring is desired, the developer should investigate and
|
|
|
|
potentially enhance the kernel perf and tracing infrastructure to export
|
2011-08-26 02:42:36 +08:00
|
|
|
GPU related performance information for consumption by performance
|
|
|
|
monitoring tools and applications.
|
2009-10-27 04:06:31 +08:00
|
|
|
</para>
|
|
|
|
</sect2>
|
|
|
|
|
|
|
|
<sect2>
|
|
|
|
<title>Configuring the device</title>
|
|
|
|
<para>
|
2011-08-26 01:18:08 +08:00
|
|
|
Obviously, device configuration is device specific.
|
2009-10-27 04:06:31 +08:00
|
|
|
However, there are several common operations: finding a
|
|
|
|
device's PCI resources, mapping them, and potentially setting
|
|
|
|
up an IRQ handler.
|
|
|
|
</para>
|
|
|
|
<para>
|
|
|
|
Finding & mapping resources is fairly straightforward. The
|
|
|
|
DRM wrapper functions, drm_get_resource_start() and
|
2011-08-26 02:46:10 +08:00
|
|
|
drm_get_resource_len(), may be used to find BARs on the given
|
2009-10-27 04:06:31 +08:00
|
|
|
drm_device struct. Once those values have been retrieved, the
|
|
|
|
driver load function can call drm_addmap() to create a new
|
2011-08-26 02:46:10 +08:00
|
|
|
mapping for the BAR in question. Note that you probably want a
|
2009-10-27 04:06:31 +08:00
|
|
|
drm_local_map_t in your driver private structure to track any
|
|
|
|
mappings you create.
|
|
|
|
<!-- !Fdrivers/gpu/drm/drm_bufs.c drm_get_resource_* -->
|
|
|
|
<!-- !Finclude/drm/drmP.h drm_local_map_t -->
|
|
|
|
</para>
|
|
|
|
<para>
|
|
|
|
if compatibility with other operating systems isn't a concern
|
|
|
|
(DRM drivers can run under various BSD variants and OpenSolaris),
|
2011-08-26 01:21:31 +08:00
|
|
|
native Linux calls may be used for the above, e.g. pci_resource_*
|
2009-10-27 04:06:31 +08:00
|
|
|
and iomap*/iounmap. See the Linux device driver book for more
|
|
|
|
info.
|
|
|
|
</para>
|
|
|
|
<para>
|
2011-08-26 01:21:31 +08:00
|
|
|
Once you have a register map, you may use the DRM_READn() and
|
2009-10-27 04:06:31 +08:00
|
|
|
DRM_WRITEn() macros to access the registers on your device, or
|
|
|
|
use driver specific versions to offset into your MMIO space
|
|
|
|
relative to a driver specific base pointer (see I915_READ for
|
2011-08-26 02:48:15 +08:00
|
|
|
an example).
|
2009-10-27 04:06:31 +08:00
|
|
|
</para>
|
|
|
|
<para>
|
|
|
|
If your device supports interrupt generation, you may want to
|
2011-08-26 02:50:14 +08:00
|
|
|
set up an interrupt handler when the driver is loaded. This
|
2009-10-27 04:06:31 +08:00
|
|
|
is done using the drm_irq_install() function. If your device
|
|
|
|
supports vertical blank interrupts, it should call
|
|
|
|
drm_vblank_init() to initialize the core vblank handling code before
|
|
|
|
enabling interrupts on your device. This ensures the vblank related
|
|
|
|
structures are allocated and allows the core to handle vblank events.
|
|
|
|
</para>
|
|
|
|
<!--!Fdrivers/char/drm/drm_irq.c drm_irq_install-->
|
|
|
|
<para>
|
2011-08-26 01:18:08 +08:00
|
|
|
Once your interrupt handler is registered (it uses your
|
2009-10-27 04:06:31 +08:00
|
|
|
drm_driver.irq_handler as the actual interrupt handling
|
|
|
|
function), you can safely enable interrupts on your device,
|
|
|
|
assuming any other state your interrupt handler uses is also
|
|
|
|
initialized.
|
|
|
|
</para>
|
|
|
|
<para>
|
|
|
|
Another task that may be necessary during configuration is
|
|
|
|
mapping the video BIOS. On many devices, the VBIOS describes
|
|
|
|
device configuration, LCD panel timings (if any), and contains
|
|
|
|
flags indicating device state. Mapping the BIOS can be done
|
|
|
|
using the pci_map_rom() call, a convenience function that
|
|
|
|
takes care of mapping the actual ROM, whether it has been
|
|
|
|
shadowed into memory (typically at address 0xc0000) or exists
|
2011-08-26 02:56:12 +08:00
|
|
|
on the PCI device in the ROM BAR. Note that after the ROM
|
|
|
|
has been mapped and any necessary information has been extracted,
|
2011-08-26 02:57:04 +08:00
|
|
|
it should be unmapped; on many devices, the ROM address decoder is
|
2011-08-26 02:57:37 +08:00
|
|
|
shared with other BARs, so leaving it mapped could cause
|
2009-10-27 04:06:31 +08:00
|
|
|
undesired behavior like hangs or memory corruption.
|
|
|
|
<!--!Fdrivers/pci/rom.c pci_map_rom-->
|
|
|
|
</para>
|
|
|
|
</sect2>
|
|
|
|
|
|
|
|
<sect2>
|
|
|
|
<title>Memory manager initialization</title>
|
|
|
|
<para>
|
|
|
|
In order to allocate command buffers, cursor memory, scanout
|
|
|
|
buffers, etc., as well as support the latest features provided
|
|
|
|
by packages like Mesa and the X.Org X server, your driver
|
|
|
|
should support a memory manager.
|
|
|
|
</para>
|
|
|
|
<para>
|
2011-08-26 01:18:08 +08:00
|
|
|
If your driver supports memory management (it should!), you
|
2010-05-28 13:33:49 +08:00
|
|
|
need to set that up at load time as well. How you initialize
|
2011-08-26 02:58:42 +08:00
|
|
|
it depends on which memory manager you're using: TTM or GEM.
|
2009-10-27 04:06:31 +08:00
|
|
|
</para>
|
|
|
|
<sect3>
|
|
|
|
<title>TTM initialization</title>
|
|
|
|
<para>
|
|
|
|
TTM (for Translation Table Manager) manages video memory and
|
|
|
|
aperture space for graphics devices. TTM supports both UMA devices
|
|
|
|
and devices with dedicated video RAM (VRAM), i.e. most discrete
|
|
|
|
graphics devices. If your device has dedicated RAM, supporting
|
2010-05-28 13:33:49 +08:00
|
|
|
TTM is desirable. TTM also integrates tightly with your
|
2009-10-27 04:06:31 +08:00
|
|
|
driver specific buffer execution function. See the radeon
|
|
|
|
driver for examples.
|
|
|
|
</para>
|
|
|
|
<para>
|
|
|
|
The core TTM structure is the ttm_bo_driver struct. It contains
|
|
|
|
several fields with function pointers for initializing the TTM,
|
|
|
|
allocating and freeing memory, waiting for command completion
|
|
|
|
and fence synchronization, and memory migration. See the
|
|
|
|
radeon_ttm.c file for an example of usage.
|
|
|
|
</para>
|
|
|
|
<para>
|
|
|
|
The ttm_global_reference structure is made up of several fields:
|
|
|
|
</para>
|
|
|
|
<programlisting>
|
|
|
|
struct ttm_global_reference {
|
|
|
|
enum ttm_global_types global_type;
|
|
|
|
size_t size;
|
|
|
|
void *object;
|
|
|
|
int (*init) (struct ttm_global_reference *);
|
|
|
|
void (*release) (struct ttm_global_reference *);
|
|
|
|
};
|
|
|
|
</programlisting>
|
|
|
|
<para>
|
|
|
|
There should be one global reference structure for your memory
|
|
|
|
manager as a whole, and there will be others for each object
|
|
|
|
created by the memory manager at runtime. Your global TTM should
|
|
|
|
have a type of TTM_GLOBAL_TTM_MEM. The size field for the global
|
|
|
|
object should be sizeof(struct ttm_mem_global), and the init and
|
|
|
|
release hooks should point at your driver specific init and
|
2011-08-26 01:18:08 +08:00
|
|
|
release routines, which probably eventually call
|
2011-08-26 03:02:52 +08:00
|
|
|
ttm_mem_global_init and ttm_mem_global_release, respectively.
|
2009-10-27 04:06:31 +08:00
|
|
|
</para>
|
|
|
|
<para>
|
|
|
|
Once your global TTM accounting structure is set up and initialized
|
2011-08-26 03:19:18 +08:00
|
|
|
by calling ttm_global_item_ref() on it,
|
2011-08-26 03:14:26 +08:00
|
|
|
you need to create a buffer object TTM to
|
2009-10-27 04:06:31 +08:00
|
|
|
provide a pool for buffer object allocation by clients and the
|
|
|
|
kernel itself. The type of this object should be TTM_GLOBAL_TTM_BO,
|
|
|
|
and its size should be sizeof(struct ttm_bo_global). Again,
|
2011-08-26 01:21:31 +08:00
|
|
|
driver specific init and release functions may be provided,
|
2011-08-26 03:19:18 +08:00
|
|
|
likely eventually calling ttm_bo_global_init() and
|
|
|
|
ttm_bo_global_release(), respectively. Also, like the previous
|
|
|
|
object, ttm_global_item_ref() is used to create an initial reference
|
2010-05-28 13:33:49 +08:00
|
|
|
count for the TTM, which will call your initialization function.
|
2009-10-27 04:06:31 +08:00
|
|
|
</para>
|
|
|
|
</sect3>
|
|
|
|
<sect3>
|
|
|
|
<title>GEM initialization</title>
|
|
|
|
<para>
|
|
|
|
GEM is an alternative to TTM, designed specifically for UMA
|
|
|
|
devices. It has simpler initialization and execution requirements
|
|
|
|
than TTM, but has no VRAM management capability. Core GEM
|
2011-08-26 03:21:17 +08:00
|
|
|
is initialized by calling drm_mm_init() to create
|
2009-10-27 04:06:31 +08:00
|
|
|
a GTT DRM MM object, which provides an address space pool for
|
2011-08-26 01:18:08 +08:00
|
|
|
object allocation. In a KMS configuration, the driver
|
|
|
|
needs to allocate and initialize a command ring buffer following
|
2011-08-26 03:42:20 +08:00
|
|
|
core GEM initialization. A UMA device usually has what is called a
|
2009-10-27 04:06:31 +08:00
|
|
|
"stolen" memory region, which provides space for the initial
|
|
|
|
framebuffer and large, contiguous memory regions required by the
|
2011-08-26 03:48:32 +08:00
|
|
|
device. This space is not typically managed by GEM, and it must
|
2009-10-27 04:06:31 +08:00
|
|
|
be initialized separately into its own DRM MM object.
|
|
|
|
</para>
|
|
|
|
<para>
|
2011-08-26 03:49:10 +08:00
|
|
|
Initialization is driver specific. In the case of Intel
|
2009-10-27 04:06:31 +08:00
|
|
|
integrated graphics chips like 965GM, GEM initialization can
|
|
|
|
be done by calling the internal GEM init function,
|
|
|
|
i915_gem_do_init(). Since the 965GM is a UMA device
|
2011-08-26 01:18:08 +08:00
|
|
|
(i.e. it doesn't have dedicated VRAM), GEM manages
|
2009-10-27 04:06:31 +08:00
|
|
|
making regular RAM available for GPU operations. Memory set
|
|
|
|
aside by the BIOS (called "stolen" memory by the i915
|
2011-08-26 01:18:08 +08:00
|
|
|
driver) is managed by the DRM memrange allocator; the
|
|
|
|
rest of the aperture is managed by GEM.
|
2009-10-27 04:06:31 +08:00
|
|
|
<programlisting>
|
|
|
|
/* Basic memrange allocator for stolen space (aka vram) */
|
|
|
|
drm_memrange_init(&dev_priv->vram, 0, prealloc_size);
|
|
|
|
/* Let GEM Manage from end of prealloc space to end of aperture */
|
|
|
|
i915_gem_do_init(dev, prealloc_size, agp_size);
|
|
|
|
</programlisting>
|
|
|
|
<!--!Edrivers/char/drm/drm_memrange.c-->
|
|
|
|
</para>
|
|
|
|
<para>
|
2011-08-26 01:21:31 +08:00
|
|
|
Once the memory manager has been set up, we may allocate the
|
2009-10-27 04:06:31 +08:00
|
|
|
command buffer. In the i915 case, this is also done with a
|
|
|
|
GEM function, i915_gem_init_ringbuffer().
|
|
|
|
</para>
|
|
|
|
</sect3>
|
|
|
|
</sect2>
|
|
|
|
|
|
|
|
<sect2>
|
|
|
|
<title>Output configuration</title>
|
|
|
|
<para>
|
2011-08-26 04:18:14 +08:00
|
|
|
The final initialization task is output configuration. This involves:
|
|
|
|
<itemizedlist>
|
|
|
|
<listitem>
|
|
|
|
Finding and initializing the CRTCs, encoders, and connectors
|
|
|
|
for the device.
|
|
|
|
</listitem>
|
|
|
|
<listitem>
|
|
|
|
Creating an initial configuration.
|
|
|
|
</listitem>
|
|
|
|
<listitem>
|
|
|
|
Registering a framebuffer console driver.
|
|
|
|
</listitem>
|
|
|
|
</itemizedlist>
|
2009-10-27 04:06:31 +08:00
|
|
|
</para>
|
|
|
|
<sect3>
|
|
|
|
<title>Output discovery and initialization</title>
|
|
|
|
<para>
|
2011-08-26 03:02:52 +08:00
|
|
|
Several core functions exist to create CRTCs, encoders, and
|
2011-08-26 04:21:22 +08:00
|
|
|
connectors, namely: drm_crtc_init(), drm_connector_init(), and
|
2009-10-27 04:06:31 +08:00
|
|
|
drm_encoder_init(), along with several "helper" functions to
|
|
|
|
perform common tasks.
|
|
|
|
</para>
|
|
|
|
<para>
|
|
|
|
Connectors should be registered with sysfs once they've been
|
|
|
|
detected and initialized, using the
|
|
|
|
drm_sysfs_connector_add() function. Likewise, when they're
|
|
|
|
removed from the system, they should be destroyed with
|
|
|
|
drm_sysfs_connector_remove().
|
|
|
|
</para>
|
|
|
|
<programlisting>
|
|
|
|
<![CDATA[
|
|
|
|
void intel_crt_init(struct drm_device *dev)
|
|
|
|
{
|
|
|
|
struct drm_connector *connector;
|
|
|
|
struct intel_output *intel_output;
|
|
|
|
|
|
|
|
intel_output = kzalloc(sizeof(struct intel_output), GFP_KERNEL);
|
|
|
|
if (!intel_output)
|
|
|
|
return;
|
|
|
|
|
|
|
|
connector = &intel_output->base;
|
|
|
|
drm_connector_init(dev, &intel_output->base,
|
|
|
|
&intel_crt_connector_funcs, DRM_MODE_CONNECTOR_VGA);
|
|
|
|
|
|
|
|
drm_encoder_init(dev, &intel_output->enc, &intel_crt_enc_funcs,
|
|
|
|
DRM_MODE_ENCODER_DAC);
|
|
|
|
|
|
|
|
drm_mode_connector_attach_encoder(&intel_output->base,
|
|
|
|
&intel_output->enc);
|
|
|
|
|
|
|
|
/* Set up the DDC bus. */
|
|
|
|
intel_output->ddc_bus = intel_i2c_create(dev, GPIOA, "CRTDDC_A");
|
|
|
|
if (!intel_output->ddc_bus) {
|
|
|
|
dev_printk(KERN_ERR, &dev->pdev->dev, "DDC bus registration "
|
|
|
|
"failed.\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
intel_output->type = INTEL_OUTPUT_ANALOG;
|
|
|
|
connector->interlace_allowed = 0;
|
|
|
|
connector->doublescan_allowed = 0;
|
|
|
|
|
|
|
|
drm_encoder_helper_add(&intel_output->enc, &intel_crt_helper_funcs);
|
|
|
|
drm_connector_helper_add(connector, &intel_crt_connector_helper_funcs);
|
|
|
|
|
|
|
|
drm_sysfs_connector_add(connector);
|
|
|
|
}
|
|
|
|
]]>
|
|
|
|
</programlisting>
|
|
|
|
<para>
|
|
|
|
In the example above (again, taken from the i915 driver), a
|
|
|
|
CRT connector and encoder combination is created. A device
|
2011-08-26 04:26:17 +08:00
|
|
|
specific i2c bus is also created for fetching EDID data and
|
2009-10-27 04:06:31 +08:00
|
|
|
performing monitor detection. Once the process is complete,
|
2011-08-26 04:26:17 +08:00
|
|
|
the new connector is registered with sysfs to make its
|
2009-10-27 04:06:31 +08:00
|
|
|
properties available to applications.
|
|
|
|
</para>
|
|
|
|
<sect4>
|
|
|
|
<title>Helper functions and core functions</title>
|
|
|
|
<para>
|
|
|
|
Since many PC-class graphics devices have similar display output
|
|
|
|
designs, the DRM provides a set of helper functions to make
|
|
|
|
output management easier. The core helper routines handle
|
2011-08-26 04:29:32 +08:00
|
|
|
encoder re-routing and the disabling of unused functions following
|
|
|
|
mode setting. Using the helpers is optional, but recommended for
|
2009-10-27 04:06:31 +08:00
|
|
|
devices with PC-style architectures (i.e. a set of display planes
|
|
|
|
for feeding pixels to encoders which are in turn routed to
|
|
|
|
connectors). Devices with more complex requirements needing
|
2011-08-26 01:21:31 +08:00
|
|
|
finer grained management may opt to use the core callbacks
|
2009-10-27 04:06:31 +08:00
|
|
|
directly.
|
|
|
|
</para>
|
|
|
|
<para>
|
|
|
|
[Insert typical diagram here.] [Insert OMAP style config here.]
|
|
|
|
</para>
|
|
|
|
</sect4>
|
|
|
|
<para>
|
2011-08-26 04:55:58 +08:00
|
|
|
Each encoder object needs to provide:
|
|
|
|
<itemizedlist>
|
|
|
|
<listitem>
|
|
|
|
A DPMS (basically on/off) function.
|
|
|
|
</listitem>
|
|
|
|
<listitem>
|
|
|
|
A mode-fixup function (for converting requested modes into
|
|
|
|
native hardware timings).
|
|
|
|
</listitem>
|
|
|
|
<listitem>
|
|
|
|
Functions (prepare, set, and commit) for use by the core DRM
|
|
|
|
helper functions.
|
|
|
|
</listitem>
|
|
|
|
</itemizedlist>
|
|
|
|
Connector helpers need to provide functions (mode-fetch, validity,
|
|
|
|
and encoder-matching) for returning an ideal encoder for a given
|
|
|
|
connector. The core connector functions include a DPMS callback,
|
|
|
|
save/restore routines (deprecated), detection, mode probing,
|
|
|
|
property handling, and cleanup functions.
|
2009-10-27 04:06:31 +08:00
|
|
|
</para>
|
|
|
|
<!--!Edrivers/char/drm/drm_crtc.h-->
|
|
|
|
<!--!Edrivers/char/drm/drm_crtc.c-->
|
|
|
|
<!--!Edrivers/char/drm/drm_crtc_helper.c-->
|
|
|
|
</sect3>
|
|
|
|
</sect2>
|
|
|
|
</sect1>
|
|
|
|
|
|
|
|
<!-- Internals: vblank handling -->
|
|
|
|
|
|
|
|
<sect1>
|
|
|
|
<title>VBlank event handling</title>
|
|
|
|
<para>
|
|
|
|
The DRM core exposes two vertical blank related ioctls:
|
2011-08-26 05:11:30 +08:00
|
|
|
<variablelist>
|
|
|
|
<varlistentry>
|
|
|
|
<term>DRM_IOCTL_WAIT_VBLANK</term>
|
|
|
|
<listitem>
|
|
|
|
<para>
|
|
|
|
This takes a struct drm_wait_vblank structure as its argument,
|
|
|
|
and it is used to block or request a signal when a specified
|
|
|
|
vblank event occurs.
|
|
|
|
</para>
|
|
|
|
</listitem>
|
|
|
|
</varlistentry>
|
|
|
|
<varlistentry>
|
|
|
|
<term>DRM_IOCTL_MODESET_CTL</term>
|
|
|
|
<listitem>
|
|
|
|
<para>
|
|
|
|
This should be called by application level drivers before and
|
|
|
|
after mode setting, since on many devices the vertical blank
|
|
|
|
counter is reset at that time. Internally, the DRM snapshots
|
|
|
|
the last vblank count when the ioctl is called with the
|
|
|
|
_DRM_PRE_MODESET command so that the counter won't go backwards
|
|
|
|
(which is dealt with when _DRM_POST_MODESET is used).
|
|
|
|
</para>
|
|
|
|
</listitem>
|
|
|
|
</varlistentry>
|
|
|
|
</variablelist>
|
2009-10-27 04:06:31 +08:00
|
|
|
<!--!Edrivers/char/drm/drm_irq.c-->
|
|
|
|
</para>
|
|
|
|
<para>
|
|
|
|
To support the functions above, the DRM core provides several
|
|
|
|
helper functions for tracking vertical blank counters, and
|
|
|
|
requires drivers to provide several callbacks:
|
|
|
|
get_vblank_counter(), enable_vblank() and disable_vblank(). The
|
|
|
|
core uses get_vblank_counter() to keep the counter accurate
|
|
|
|
across interrupt disable periods. It should return the current
|
|
|
|
vertical blank event count, which is often tracked in a device
|
|
|
|
register. The enable and disable vblank callbacks should enable
|
|
|
|
and disable vertical blank interrupts, respectively. In the
|
|
|
|
absence of DRM clients waiting on vblank events, the core DRM
|
2011-08-26 01:18:08 +08:00
|
|
|
code uses the disable_vblank() function to disable
|
|
|
|
interrupts, which saves power. They are re-enabled again when
|
2009-10-27 04:06:31 +08:00
|
|
|
a client calls the vblank wait ioctl above.
|
|
|
|
</para>
|
|
|
|
<para>
|
2011-08-26 03:41:37 +08:00
|
|
|
A device that doesn't provide a count register may simply use an
|
2009-10-27 04:06:31 +08:00
|
|
|
internal atomic counter incremented on every vertical blank
|
2011-08-26 03:41:37 +08:00
|
|
|
interrupt (and then treat the enable_vblank() and disable_vblank()
|
|
|
|
callbacks as no-ops).
|
2009-10-27 04:06:31 +08:00
|
|
|
</para>
|
|
|
|
</sect1>
|
|
|
|
|
|
|
|
<sect1>
|
|
|
|
<title>Memory management</title>
|
|
|
|
<para>
|
2011-08-26 02:10:12 +08:00
|
|
|
The memory manager lies at the heart of many DRM operations; it
|
|
|
|
is required to support advanced client features like OpenGL
|
2011-08-26 02:58:42 +08:00
|
|
|
pbuffers. The DRM currently contains two memory managers: TTM
|
2009-10-27 04:06:31 +08:00
|
|
|
and GEM.
|
|
|
|
</para>
|
|
|
|
|
|
|
|
<sect2>
|
|
|
|
<title>The Translation Table Manager (TTM)</title>
|
|
|
|
<para>
|
|
|
|
TTM was developed by Tungsten Graphics, primarily by Thomas
|
|
|
|
Hellström, and is intended to be a flexible, high performance
|
|
|
|
graphics memory manager.
|
|
|
|
</para>
|
|
|
|
<para>
|
|
|
|
Drivers wishing to support TTM must fill out a drm_bo_driver
|
|
|
|
structure.
|
|
|
|
</para>
|
|
|
|
<para>
|
|
|
|
TTM design background and information belongs here.
|
|
|
|
</para>
|
|
|
|
</sect2>
|
|
|
|
|
|
|
|
<sect2>
|
|
|
|
<title>The Graphics Execution Manager (GEM)</title>
|
|
|
|
<para>
|
|
|
|
GEM is an Intel project, authored by Eric Anholt and Keith
|
|
|
|
Packard. It provides simpler interfaces than TTM, and is well
|
|
|
|
suited for UMA devices.
|
|
|
|
</para>
|
|
|
|
<para>
|
|
|
|
GEM-enabled drivers must provide gem_init_object() and
|
|
|
|
gem_free_object() callbacks to support the core memory
|
|
|
|
allocation routines. They should also provide several driver
|
|
|
|
specific ioctls to support command execution, pinning, buffer
|
|
|
|
read & write, mapping, and domain ownership transfers.
|
|
|
|
</para>
|
|
|
|
<para>
|
|
|
|
On a fundamental level, GEM involves several operations: memory
|
|
|
|
allocation and freeing, command execution, and aperture management
|
|
|
|
at command execution time. Buffer object allocation is relatively
|
|
|
|
straightforward and largely provided by Linux's shmem layer, which
|
|
|
|
provides memory to back each object. When mapped into the GTT
|
|
|
|
or used in a command buffer, the backing pages for an object are
|
|
|
|
flushed to memory and marked write combined so as to be coherent
|
|
|
|
with the GPU. Likewise, when the GPU finishes rendering to an object,
|
|
|
|
if the CPU accesses it, it must be made coherent with the CPU's view
|
|
|
|
of memory, usually involving GPU cache flushing of various kinds.
|
|
|
|
This core CPU<->GPU coherency management is provided by the GEM
|
|
|
|
set domain function, which evaluates an object's current domain and
|
|
|
|
performs any necessary flushing or synchronization to put the object
|
|
|
|
into the desired coherency domain (note that the object may be busy,
|
2011-08-26 02:57:04 +08:00
|
|
|
i.e. an active render target; in that case, the set domain function
|
2011-08-26 01:18:08 +08:00
|
|
|
blocks the client and waits for rendering to complete before
|
2009-10-27 04:06:31 +08:00
|
|
|
performing any necessary flushing operations).
|
|
|
|
</para>
|
|
|
|
<para>
|
|
|
|
Perhaps the most important GEM function is providing a command
|
|
|
|
execution interface to clients. Client programs construct command
|
|
|
|
buffers containing references to previously allocated memory objects
|
2011-08-26 01:18:08 +08:00
|
|
|
and submit them to GEM. At that point, GEM takes care to bind
|
2009-10-27 04:06:31 +08:00
|
|
|
all the objects into the GTT, execute the buffer, and provide
|
|
|
|
necessary synchronization between clients accessing the same buffers.
|
|
|
|
This often involves evicting some objects from the GTT and re-binding
|
|
|
|
others (a fairly expensive operation), and providing relocation
|
|
|
|
support which hides fixed GTT offsets from clients. Clients must
|
|
|
|
take care not to submit command buffers that reference more objects
|
|
|
|
than can fit in the GTT or GEM will reject them and no rendering
|
|
|
|
will occur. Similarly, if several objects in the buffer require
|
|
|
|
fence registers to be allocated for correct rendering (e.g. 2D blits
|
|
|
|
on pre-965 chips), care must be taken not to require more fence
|
|
|
|
registers than are available to the client. Such resource management
|
|
|
|
should be abstracted from the client in libdrm.
|
|
|
|
</para>
|
|
|
|
</sect2>
|
|
|
|
|
|
|
|
</sect1>
|
|
|
|
|
|
|
|
<!-- Output management -->
|
|
|
|
<sect1>
|
|
|
|
<title>Output management</title>
|
|
|
|
<para>
|
|
|
|
At the core of the DRM output management code is a set of
|
2011-08-26 03:02:52 +08:00
|
|
|
structures representing CRTCs, encoders, and connectors.
|
2009-10-27 04:06:31 +08:00
|
|
|
</para>
|
|
|
|
<para>
|
|
|
|
A CRTC is an abstraction representing a part of the chip that
|
|
|
|
contains a pointer to a scanout buffer. Therefore, the number
|
|
|
|
of CRTCs available determines how many independent scanout
|
|
|
|
buffers can be active at any given time. The CRTC structure
|
|
|
|
contains several fields to support this: a pointer to some video
|
|
|
|
memory, a display mode, and an (x, y) offset into the video
|
|
|
|
memory to support panning or configurations where one piece of
|
|
|
|
video memory spans multiple CRTCs.
|
|
|
|
</para>
|
|
|
|
<para>
|
|
|
|
An encoder takes pixel data from a CRTC and converts it to a
|
|
|
|
format suitable for any attached connectors. On some devices,
|
|
|
|
it may be possible to have a CRTC send data to more than one
|
|
|
|
encoder. In that case, both encoders would receive data from
|
|
|
|
the same scanout buffer, resulting in a "cloned" display
|
|
|
|
configuration across the connectors attached to each encoder.
|
|
|
|
</para>
|
|
|
|
<para>
|
|
|
|
A connector is the final destination for pixel data on a device,
|
|
|
|
and usually connects directly to an external display device like
|
|
|
|
a monitor or laptop panel. A connector can only be attached to
|
|
|
|
one encoder at a time. The connector is also the structure
|
|
|
|
where information about the attached display is kept, so it
|
|
|
|
contains fields for display data, EDID data, DPMS &
|
|
|
|
connection status, and information about modes supported on the
|
|
|
|
attached displays.
|
|
|
|
</para>
|
|
|
|
<!--!Edrivers/char/drm/drm_crtc.c-->
|
|
|
|
</sect1>
|
|
|
|
|
|
|
|
<sect1>
|
|
|
|
<title>Framebuffer management</title>
|
|
|
|
<para>
|
|
|
|
In order to set a mode on a given CRTC, encoder and connector
|
|
|
|
configuration, clients need to provide a framebuffer object which
|
2011-08-26 01:18:08 +08:00
|
|
|
provides a source of pixels for the CRTC to deliver to the encoder(s)
|
2009-10-27 04:06:31 +08:00
|
|
|
and ultimately the connector(s) in the configuration. A framebuffer
|
|
|
|
is fundamentally a driver specific memory object, made into an opaque
|
|
|
|
handle by the DRM addfb function. Once an fb has been created this
|
|
|
|
way it can be passed to the KMS mode setting routines for use in
|
|
|
|
a configuration.
|
|
|
|
</para>
|
|
|
|
</sect1>
|
|
|
|
|
|
|
|
<sect1>
|
|
|
|
<title>Command submission & fencing</title>
|
|
|
|
<para>
|
|
|
|
This should cover a few device specific command submission
|
|
|
|
implementations.
|
|
|
|
</para>
|
|
|
|
</sect1>
|
|
|
|
|
|
|
|
<sect1>
|
|
|
|
<title>Suspend/resume</title>
|
|
|
|
<para>
|
|
|
|
The DRM core provides some suspend/resume code, but drivers
|
|
|
|
wanting full suspend/resume support should provide save() and
|
2011-08-26 01:18:08 +08:00
|
|
|
restore() functions. These are called at suspend,
|
2009-10-27 04:06:31 +08:00
|
|
|
hibernate, or resume time, and should perform any state save or
|
|
|
|
restore required by your device across suspend or hibernate
|
|
|
|
states.
|
|
|
|
</para>
|
|
|
|
</sect1>
|
|
|
|
|
|
|
|
<sect1>
|
|
|
|
<title>DMA services</title>
|
|
|
|
<para>
|
|
|
|
This should cover how DMA mapping etc. is supported by the core.
|
|
|
|
These functions are deprecated and should not be used.
|
|
|
|
</para>
|
|
|
|
</sect1>
|
|
|
|
</chapter>
|
|
|
|
|
|
|
|
<!-- External interfaces -->
|
|
|
|
|
|
|
|
<chapter id="drmExternals">
|
|
|
|
<title>Userland interfaces</title>
|
|
|
|
<para>
|
|
|
|
The DRM core exports several interfaces to applications,
|
|
|
|
generally intended to be used through corresponding libdrm
|
|
|
|
wrapper functions. In addition, drivers export device specific
|
|
|
|
interfaces for use by userspace drivers & device aware
|
|
|
|
applications through ioctls and sysfs files.
|
|
|
|
</para>
|
|
|
|
<para>
|
|
|
|
External interfaces include: memory mapping, context management,
|
|
|
|
DMA operations, AGP management, vblank control, fence
|
|
|
|
management, memory management, and output management.
|
|
|
|
</para>
|
|
|
|
<para>
|
|
|
|
Cover generic ioctls and sysfs layout here. Only need high
|
2011-08-26 01:18:08 +08:00
|
|
|
level info, since man pages should cover the rest.
|
2009-10-27 04:06:31 +08:00
|
|
|
</para>
|
|
|
|
</chapter>
|
|
|
|
|
|
|
|
<!-- API reference -->
|
|
|
|
|
|
|
|
<appendix id="drmDriverApi">
|
|
|
|
<title>DRM Driver API</title>
|
|
|
|
<para>
|
|
|
|
Include auto-generated API reference here (need to reference it
|
|
|
|
from paragraphs above too).
|
|
|
|
</para>
|
|
|
|
</appendix>
|
|
|
|
|
|
|
|
</book>
|