110 lines
4.3 KiB
Plaintext
110 lines
4.3 KiB
Plaintext
Reference counting in pnfs:
|
|
==========================
|
|
|
|
The are several inter-related caches. We have layouts which can
|
|
reference multiple devices, each of which can reference multiple data servers.
|
|
Each data server can be referenced by multiple devices. Each device
|
|
can be referenced by multiple layouts. To keep all of this straight,
|
|
we need to reference count.
|
|
|
|
|
|
struct pnfs_layout_hdr
|
|
----------------------
|
|
The on-the-wire command LAYOUTGET corresponds to struct
|
|
pnfs_layout_segment, usually referred to by the variable name lseg.
|
|
Each nfs_inode may hold a pointer to a cache of these layout
|
|
segments in nfsi->layout, of type struct pnfs_layout_hdr.
|
|
|
|
We reference the header for the inode pointing to it, across each
|
|
outstanding RPC call that references it (LAYOUTGET, LAYOUTRETURN,
|
|
LAYOUTCOMMIT), and for each lseg held within.
|
|
|
|
Each header is also (when non-empty) put on a list associated with
|
|
struct nfs_client (cl_layouts). Being put on this list does not bump
|
|
the reference count, as the layout is kept around by the lseg that
|
|
keeps it in the list.
|
|
|
|
deviceid_cache
|
|
--------------
|
|
lsegs reference device ids, which are resolved per nfs_client and
|
|
layout driver type. The device ids are held in a RCU cache (struct
|
|
nfs4_deviceid_cache). The cache itself is referenced across each
|
|
mount. The entries (struct nfs4_deviceid) themselves are held across
|
|
the lifetime of each lseg referencing them.
|
|
|
|
RCU is used because the deviceid is basically a write once, read many
|
|
data structure. The hlist size of 32 buckets needs better
|
|
justification, but seems reasonable given that we can have multiple
|
|
deviceid's per filesystem, and multiple filesystems per nfs_client.
|
|
|
|
The hash code is copied from the nfsd code base. A discussion of
|
|
hashing and variations of this algorithm can be found at:
|
|
http://groups.google.com/group/comp.lang.c/browse_thread/thread/9522965e2b8d3809
|
|
|
|
data server cache
|
|
-----------------
|
|
file driver devices refer to data servers, which are kept in a module
|
|
level cache. Its reference is held over the lifetime of the deviceid
|
|
pointing to it.
|
|
|
|
lseg
|
|
----
|
|
lseg maintains an extra reference corresponding to the NFS_LSEG_VALID
|
|
bit which holds it in the pnfs_layout_hdr's list. When the final lseg
|
|
is removed from the pnfs_layout_hdr's list, the NFS_LAYOUT_DESTROYED
|
|
bit is set, preventing any new lsegs from being added.
|
|
|
|
layout drivers
|
|
--------------
|
|
|
|
PNFS utilizes what is called layout drivers. The STD defines 3 basic
|
|
layout types: "files" "objects" and "blocks". For each of these types
|
|
there is a layout-driver with a common function-vectors table which
|
|
are called by the nfs-client pnfs-core to implement the different layout
|
|
types.
|
|
|
|
Files-layout-driver code is in: fs/nfs/nfs4filelayout.c && nfs4filelayoutdev.c
|
|
Objects-layout-deriver code is in: fs/nfs/objlayout/.. directory
|
|
Blocks-layout-deriver code is in: fs/nfs/blocklayout/.. directory
|
|
|
|
objects-layout setup
|
|
--------------------
|
|
|
|
As part of the full STD implementation the objlayoutdriver.ko needs, at times,
|
|
to automatically login to yet undiscovered iscsi/osd devices. For this the
|
|
driver makes up-calles to a user-mode script called *osd_login*
|
|
|
|
The path_name of the script to use is by default:
|
|
/sbin/osd_login.
|
|
This name can be overridden by the Kernel module parameter:
|
|
objlayoutdriver.osd_login_prog
|
|
|
|
If Kernel does not find the osd_login_prog path it will zero it out
|
|
and will not attempt farther logins. An admin can then write new value
|
|
to the objlayoutdriver.osd_login_prog Kernel parameter to re-enable it.
|
|
|
|
The /sbin/osd_login is part of the nfs-utils package, and should usually
|
|
be installed on distributions that support this Kernel version.
|
|
|
|
The API to the login script is as follows:
|
|
Usage: $0 -u <URI> -o <OSDNAME> -s <SYSTEMID>
|
|
Options:
|
|
-u target uri e.g. iscsi://<ip>:<port>
|
|
(allways exists)
|
|
(More protocols can be defined in the future.
|
|
The client does not interpret this string it is
|
|
passed unchanged as received from the Server)
|
|
-o osdname of the requested target OSD
|
|
(Might be empty)
|
|
(A string which denotes the OSD name, there is a
|
|
limit of 64 chars on this string)
|
|
-s systemid of the requested target OSD
|
|
(Might be empty)
|
|
(This string, if not empty is always an hex
|
|
representation of the 20 bytes osd_system_id)
|
|
|
|
blocks-layout setup
|
|
-------------------
|
|
|
|
TODO: Document the setup needs of the blocks layout driver
|