![]() |
![]() |
|
|
![]() |
![]() |
HP-UX MultiProcessing: White Paper > Chapter 1 MultiProcessing![]() Spinlocks |
|
Spinlocks are at the heart of controlling concurrency within an MP system. Their chief purpose is to protect global data structures by controlling access to critical data. When entering an area of code that modifies a global data structure, the kernel acquires an associated spinlock and then releases it when leaving the affected area of code. Spinlocks are a more fundamental way of protecting critical sections than semaphores, in that they are used in the construction of the semaphore services; semaphore implementations themselves are critical sections.
The spinlock routines operate on a binary flag of type lock_t, to guarantee mutual exclusion of threads of control. The functionality of spinlock/spinunlock to raise the spl priority to mask out external interrupts and prevent preemption: old_priority = raise_priority (HIGHEST_PRIORITY); while (test_and_clear (lock) == 0); ...U: lock = 1; restore_priority (old_priority); To avert deadlocks, spinlock acquisition enforces a simple ordering constraint: Do not attempt to lock a lower or equal-order spinlock to one already held. The following rules govern use of spinlocks:
Numerous spinlocks are created at the time of kernel initialization with a call to alloc_spinlock(), which primarily allocates memory for the spinlock data structure and initializes its fields. The kernel creates these spinlocks from init_spinlocks() and by calls to vm_initlock() for the VM spinlocks. The table below lists some of the spinlocks allocated at the time of kernel initialization. Other spinlocks are created and destroyed during runtime. Table 1-3 Spinlocks allocated when kernel is initialized
To improve the performance of the spinlock code, HP-UX implements a technique called "dynamic inlining." A macro is used for select performance-sensitive spinlocks that reserves space for inlining the spinlock instead of simply calling the spinlock function. This is done at compile time. At execution time, if the system has more than one processor, the macro is replaced with inline spinlock code. For systems with more than one processor, the mutual exclusion algorithm now uses an LDCW instruction, which reduces the pathlength of the spinlock routines. HP-UX uses two types of data structures for its spinlock implementation:
There is one lock_t data structure for every spinlock. The table that follows describes the elements in the structure. Table 1-4 Elements of the spinlock data structure lock_t
A single spinlock works well for a single instance of a global data structure or one that is accessed in synchronously. However, contention occurs when using a single spinlock for a data structure with multiple instances (such as a vnode structure). Conversely, using a single spinlock for each vnode would be overcompensating. To compromise, HP-UX allocates the capability to use a pool of "hashed spinlocks" that are accessed by a hash function to deal with data structures having multiple instances of individual entries or a group of entries. When developing code, a programmer can choose to do a hash for a hash pool covering any particular requirement (for example, one for vnodes, one for inodes, etc). The routine alloc_h_spinlock_pool() is used to allocate a pool of hashed spinlocks. From this routine, the kernel calls from init_hashed_spinlocks(). A spinlock for a particular instance is then accessed by hashing on the address or some other unique attribute of that instance. You can see spinlocks obtained through this hash pool by a call to MP_H_SPINLOCK() in the kernel. Some of the hashed pools currently allocated by the kernel are
Table 1-5 Key elements in hash_sl_pool structure
The hash functions return an index into this array of pointers. Regardless of whether the spinlock data structure is accessed directly or through a hash table, the acquisition details to be discussed next are the same. To ensure that no processor is kept waiting indefinitely for a spinlock, round-robin arbitration using two modules takes place. Table 1-6 Modules for spinlock arbitration
|
![]() |
||
![]() |
![]() |
![]() |
|||||||||
|