windows-nt/Source/XPSP1/NT/admin/netui/macprint/spooler/cdd.txt
2020-09-26 16:20:57 +08:00

106 lines
4.5 KiB
Plaintext

MACPRINT SPOOLER CDD
Design Goals:
MacPrint is ported from the OS/2 implementation. The primary design
goal is to simplify the porting process as much as possible. The
original structure of MacPrint was retained and modified only where
necessary. There are 3 main areas of modification:
* configuration
* synchronization
* removal of the despooler component
* support for cached dictionaries
configuration changes -
Where the OS/2 implementation depended on being able to determine
at service start time the configuration for all possible shared
print queues (this was read from lanman.ini), the NT implementation
allows for the dynamic addition and removal of printer objects
without specifying configuration information in a data file. This
forced the modification of the main queue list structure to be
a single linked list instead of an array. In the NT implementation,
there is now a single data structure for NT printer objects, and
being in this structure implies the existence of a validly configured
NT Printer Object.
All of the code to read and parse configuration information from
LANMAN.INI at service start time was removed and replaced by code
to get configuration information from the EnumPrinters() API as new
printers are discovered.
synchronization changes -
NT provides a more robust synchronization mechanism than OS/2.
WaitForMultipleObjects() actually works correctly under NT, and
it is possible to wait on objects besides semaphores. Mutual
exclusion to data is achieved through critical sections as opposed
to mutexes or semaphores. Critical sections require fewer resources
than either, but can only be shared between threads of the same
process. Events are used to process jobs asynchronously. Instead
of clearing a semaphore when a read or open occurs, an event is
signaled. Semaphores in NT are used as a gating mechanism,
and are not designed to be used as a signaling mechanism. Syncronization
with the termination of threads is achieved by waiting on the thread
object as opposed to waiting on an event that the thread signals
just before it terminates.
removal of the despooler -
The spooler and despooler were somewhat integrated in the OS/2
implementation - especially with regards to configuration information.
Data structures pertinent to the despooler only were removed, and
the configuration routines are no longer contained in a DLL shared
by the spooler and despooler. The only connection between the two
components is now through a PostScript comment that is included by
the spooler to indicate if ctrl-d's should be stripped by the
despooler.
support for cached dictionaries -
As the caching of PostScript dictionaries achieved only modest performance
improvements at best, and since it added significant complexity to the
code, support for cached dictionaries will most likely be removed in
the NT implementation. If at coding time it becomes clear that more
work is involved to remove the feature than to port it, it may stay
in.
Thread/Process Structure:
MacPrint is an NT Service, hence it follows the NT Service Model. main()
is the service control dispatcher for the MacPrint service. This thread
is used by the NT Service Controller to communicate with the MacPrint
Service.
MacPrintMain() is the main thread for MacPrint. It is responsible for
monitoring the NT Print Manager for configuration changes (creation
and removal of printer objects) and for starting and stopping threads
to manage each NT Printer Object. Essentially, MacPrintMain polls
until a service stop request is received. While polling, it enumerates
printers from the NT Print Manager and either creates a control thread
for a newly discovered printer or terminates the control thread for
each printer no longer configured.
QueueServiceThread() is the thread for each active NT Printer Object.
This thread listens for jobs directed to the NT printer and processes
the job. One thread supports multiple jobs for the printer. This is
done by using two events per queue - ReadEvent and OpenEvent. The
PAP API have the property that they take an address to store a completion
code when the event completes, and this address is specified to be in
a per job data structure. The event is common for all jobs in the
queue. When a particular event is signaled, the list of pending jobs
in that queue is walked to find one that is in the correct state
(read_pending or open_pending) and that has it's completion code set.
The event is processed for that job, and the job is moved to the end
of the list so that events for jobs are processed in round-robin
order.