Index Index for
Section 4
Index Alphabetical
listing for C
Bottom of page Bottom of
page

class_scheduling(4)

NAME

class_scheduling, class.h - Allocate CPU resources based on scheduling classes

SYNOPSIS

#include <sys/class.h>

DESCRIPTION

The class scheduling features included in the operating system software let you restrict the percentage of CPU time allowed different kinds of users and tasks. This helps you allocate CPU resources so that the most important work receives the processing time it requires. For example, you might want to run two versions of a production database on your system. One version is used as part of your business operations, while the other is a test copy, with different tuning parameters. The test database can be assigned to a class that has a lower percentage of CPU time so that your daily operations are not impacted by the testing. As another example, you might want to give background daemons, such as the print spooler, less access time to CPUs to improve response time for interactive users. By using the class_admin utility, you can: · Create and maintain one or more databases of scheduling classes, each of which has a specified percentage of CPU time. The databases are typically stored in the /etc/class directory and cannot be edited manually. You can set up different class databases for different time intervals during a 24-hour period (for example, one database that favors interactive users during the day and another that favors batch jobs at night). A class_admin load command can then be included as a cron job to load the appropriate database into shared memory at a certain time. · Assign members to different classes. Members assigned to a particular class can be users, groups, processes, process groups, sessions, or some combination of these. · Enable and disable the daemon that schedules execution of different tasks according to what you've defined in the database currently loaded into memory. Among the configuration parameters you supply during setup of the class database is a time interval that controls how often the daemon refreshes the structure used by the kernel to implement your class scheduling priorities. You cannot use the class_admin utility to add application names to a class. However, after defining a class in the current database and enabling class scheduling, you can use runclass commands to launch programs in that class. When you do this, identifiers for all the application processes are automatically added as members of the specified class. The runclass command is particularly useful as a cron job that executes after the one that enables class scheduling and loads a particular class database into memory. The libclass library provides programming routines for setting up and using class scheduling. The name for each routine in this library uses the format class_operation, for example class_create(). A complete list of routines is provided in the SEE ALSO section. How Class Scheduling Works The kernel has very little internal knowledge of class scheduling. Much of the work done by the class scheduler is done in user space. To the kernel, a class is simply an element in an array of integers, each specifying a number of clock ticks that are available for a certain time interval. The time interval is set when the class scheduling database is configured, and the integer values vary, depending on the CPU percentage specified for the different classes. A thread that is subject to class scheduling has an index into the array. Each time the thread uses CPU time, the kernel decrements the number of clock ticks used from the array element. When the count reaches zero, the thread is prevented from running. A class database can be configured to impose CPU percentages in either a hard or soft way. Hard percentages (the default) mean that, once a thread uses its allotted percentage of CPU time, it is prevented from running until the next time interval. Soft percentages mean that thread execution can continue after the CPU percentage is reached if the system has idle CPUs (if there is no contention for CPU time among users). Hard percentages risk wasting CPU cycles; however, they are appropriate for a system on which users contract and pay for a certain allocation of system resources.. When the class scheduler daemon is started, it loads the database into memory, creates an initial array of clock-tick integers used by the kernel, and writes this array to the kernel. Along with the array, the daemon gives the kernel the time interval during which integers in the array are to be decremented. Then the daemon alternately sleeps and wakes up to refresh the array. The daemon calculates the total number of clock ticks in the interval during which it sleeps by using the following formula: interval_in_seconds * clock_ticks_per_second * number_of_CPUs The deamon then allocates the total number of clock ticks among classes according to the specified percentages, creates the array of integers, and writes the array to the kernel. Should the daemon terminate or fail to wake up, the kernel automatically disables class scheduling after twice the specified time interval has passed. If this happens, processes belonging to scheduling classes then run freely, no longer subject to their CPU percentage restrictions. Class scheduling always operates in the context of a soft partition, which is currently mapped to a processor set. (See processor_sets(4).) This means that "number_of_CPUs" in the preceding formula represents the number of CPUs in the partition associated with the class database. If user-defined partitions do not exist on the system, all system CPUs belong to the default partition (processor set 0), and the system can have only one active database and one instance of a class scheduler daemon. If system CPUs have been allocated into different partitions, you have the option of setting up class scheduling on one or more of the available partitions. By default, the class_admin utility operates on the system default partition; however, the utility includes the setp subcommand to change partition context for subsequent subcommands. For each partition where you want class scheduling to be implemented, you must create at least one class scheduling database and enable class scheduling. See class_admin(8) for details. A process may be represented in a class database by members in more than one class. For example, the group ID representing the group staff might be a member of class A and the user ID representing the user guest might be a member of class B. If user guest belongs to group staff, the user belongs to two classes. In such cases, the class containing the most restrictive type of ID applies to the process. For the example just described, user ID is more restrictive than group ID, so the process for user guest would execute in class B. When member types are ranked from least to most restrictive, the order is as follows: group ID, user ID, session ID, process group ID, process ID. Class scheduling is subject to the following limits, as defined in the class.h header file: · Maximum of partitions (class database/daemon pairs) per system: 100 · Maximum number of classes per database: 100 · Maximum number of members across all classes: 2500 · Maximum number of characters in a class database pathname (for save and load subcommands): 80 · Maximum number of characters in a class name: 20 Class scheduling is subject to the following restrictions: · You must be superuser to invoke the class_admin utility. · Users cannot use the runclass command to raise an application's priority above what it would otherwise have through definitions in the class database. In other words, a user who is a member of a class entitled to a certain percentage of CPU time cannot use the runclass command to start applications at a higher priority than what the user is entitled to. · For most operations that applications perform by using routines in the libclass library, root permission is required. Using Class Scheduling To implement class scheduling, you must first create a database file and populate the file with one or more classes. You assign each class a percentage of the total CPU time available. One or more applications or groups of applications can be assigned to a class, specified by the appropriate identifying number, such as a GID, UID, or PID. Any identifiers that are temporary, such as a PID, do not persist across a reboot and cease to exist when a task is completed. Therefore, they have no effect when the system or task is restarted. For this reason, the members you add to the database when first creating it are usually user and group identifiers. After the database is established, you can start the class scheduling daemon to put the CPU access restrictions into effect. The class_admin utility has subcommands that let you review classes, change members and CPU percentages, delete members or entire classes, and review assigned CPU percentages against actual use. You can invoke the utility either interactively or by using a script. Once a class scheduling database is configured and its corresponding daemon enabled, you can start applications in any of the existing classes by using the runclass command. This allows you to indirectly add entirely new processes to a class. It also allows you to temporarily lower or (if you have superuser privilege) increase the CPU percentages for processes that otherwise startup in a class with a different CPU percentage. For example, you might set a value for interactive operations that is much higher than background processes such as print daemons. If it is later necessary to temporarily use the higher value for a particular print job, you can use the runclass command to execute the lpd daemon in the same class as interactive operations. After the print job has completed, the class_admin delete subcommand can be used to remove the process identifier for lpd from the class. The following sections suggest a systematic approach to using class scheduling, although you can use it equally well to quickly fix a CPU resource sharing problem. Planning CPU Resource Allocation How you allocate CPU resources will depend on your system environment and which resources and priorities must be considered. A typical scenario is to assign a higher priority to interactive tasks so that users do not encounter long response times. Most batch or background processes will be assigned a lower priority, while some specific background processes may require a higher priority. For example, if a nightly backup is being performed, you might not want it to have such a low priority that it does not complete in a reasonable time. An alternative scenario is if there are critical realtime tasks, such as in process-control applications, that should take priority over interactive processes. For this scenario, you should design a baseline, assigning processes to classes and then monitor tasks and user feedback to tune the database by moving tasks from class to class or changing the CPU access time of the classes. While thinking about CPU resource allocation, you need to decide whether you should use class scheduling, a user-defined processor set, or both on your system. CPUs in a user-defined processor set cannot be used by any process that is not explicitly run on that processor set. Sometimes reserving specific CPUs for use only by certain applications, such as those doing realtime process control, is warranted. For critical applications, a processor set can provide a better guarantee of immediate CPU availability than class scheduling if the system load is high and load patterns are not always predictable. When running selected applications in a non-default processor set, you trade off potential waste of CPU resources to guarantee that CPUs are available for the applications running in that processor set. Setting "soft" CPU percentages for scheduling classes allows the kernel to apply only the idle CPUs in the processor set for which the class database was created. The kernel cannot use idle CPUs in one processor set to ease CPU contention by processes running in another processor set. This means that the performance of less critical applications might not be acceptable after they are restricted to the reduced number of CPUs in the default processor set. Using class scheduling in a processor set to better control CPU access is not likely to help when there are simply too few CPUs to do the required amount of work. Sometimes your site might be running an application that incorrectly spawns large numbers of threads that immediately grab more CPU resources than the application needs. Until the application can be corrected, you can define a new processor set, populate it with a subset of the system CPUs, and use the runon command to start the application in the new processor set. In this case, it might or might not be worthwhile to use class scheduling in the default processor set where remaining applications are run and interactive user processes are running. If your intention is simply to apply a more appropriate allocation of CPU resources than the kernel applies by default, try class scheduling by itself (on the default processor set while it contains all system CPUs) to see if this approach accomplishes what you want. Consider defining and using an additional processor set in order to meet the requirements of applications with unusual requirements. Steps for Setting Up and Using Class Scheduling The usual process for setting up class scheduling is as follows: 1. Decide how you want to allocate the CPU resources (decide on class groupings of users and tasks). 2. Use the class_admin utility's subcommands to set up and maintain the class database: · Configure the database. For this step, you specify: whether you want a default class for processes (other than those for UID 0) that are not members of the classes you define, whether you want hard or soft enforcement of CPU percentages, and how frequently the scheduler checks CPU usage by classes. · Create classes and add any users and groups to classes by using the create and add subcommands. When creating classes, keep in mind the 20-character restriction on class names and the fact that CPU percentages for all your classes cannot add up to more than 100 percent. · Verify class entries with the show subcommand. · Use the save subcommand to write database entries to /etc/class/filename. Keep in mind the 80-character pathname restriction if you prefer your own directory locations and names for class databases. · Use the enable subcommand to start the scheduler. Applications are usually added to classes after class scheduling is enabled by using the runclass command, which is not a subcommand in the class_admin utility. · Use the stats subcommand to check target and actual CPU usages for the different classes. The following example shows an interactive class_admin session that sets up two class databases (daytime and nighttime) for the default partition. In this example, both databases have the same two classes (interactive_users and batch_jobs), however, the CPU percentages for these classes and the time interval for resetting class usage are set differently. After the two databases are created and saved to disk, the daytime database is loaded into memory, the scheduler daemon is enabled, and runtime statistics are checked: # class_admin Class Scheduler Administration configure: Shall processes that have not been explicitly assigned to a defined class be assigned to a 'default' class? Enter (yes/no) [no]: Enforce class scheduling when the CPU is otherwise idle? (yes/no) [yes]: no How often do you want the system to reset class usage? Enter number of seconds (1): class> show Configuration: -Processes not explicitly defined in the database are not class scheduled. -If the processor has some idle time, class scheduled processes are allowed to exceed their cpu percentage. -The class scheduler will check class CPU usage every 1 seconds. current partition: 0 current database: /etc/class/part.default Class scheduler status: disabled classes: class> create interactive_users 50 interactive_users created at 50% cpu usage class> create batch_jobs 10 batch_jobs created at 10% cpu usage class> add interactive_users uid 234 457 235 uid 234 457 235 added to interactive_users. class> save /etc/class/daytime database /etc/class/daytime saved class> modify interactive_users 10 interactive_users targeted at 10% usage class> modify batch_jobs 50 batch_jobs targeted at 50% class> configure Shall processes that have not been explicitly assigned to a defined class be assigned to a 'default' class? Enter (yes/no) [no]: Enforce class scheduling when the CPU is otherwise idle? (yes/no) [yes]: no How often do you want the system to reset class usage? Enter number of seconds (1): 5 class> save /etc/class/nighttime database /etc/class/nighttime saved class> load /etc/class/daytime The following warning and prompt is displayed because this session has not saved any changes to the /etc/class/part.default file, which is the database to which context was set at the beginning of the interactive session. However, all permanent changes were saved to databases with more meaningful names, and there is no intention to put the /etc/class/part.default database into use. Therefore, the utility is told to continue with the load operation: current database modified and not saved load new database anyway? (yes/no) [yes]: yes database /etc/class/daytime loaded class> enable class scheduler enabled class> In another terminal window, the runclass command is used to start one or two low-priority applications in the batch_jobs class. For example: # runclass batch_jobs program_name Then, runtime statistics can be checked in the window of the class_admin session: class> stats Class scheduler status: enabled class name target percentage actual percentage interactive_users 50% 40.0% batch_jobs 10% 9.0% class> While the database is loaded and class scheduling enabled, you can modify the database dynamically to achieve the results you want by using the configure, change, add, and destroy subcommands, interspersed with the show and stats subcommands to check results. Enter a final save /etc/class/filename subcommand when you are satisfied with your modifications. When run in interactive mode, the class_admin utility asks on exit or quit whether you want to write database changes to disk if you have not explicitly saved them during your session. If the utility is run from a script, any changes to the class database are saved to disk automatically. The changes that are saved to disk do not include changes in class membership for temporary identifiers (process IDs, process group IDs, and session IDs) because these do not persist across system reboots. It is always a good idea to do explicit save operations when you intend to create or modify class databases that have non-default names or that are stored in directories other than /etc/class. You also have the programming option of using the class_* routines in the libclass library to do the operations discussed in this section.

FILES

/etc/class/part.default Default directory and name for the class database in the system default partition (processor set 0). /etc/class/part.pset_number Default directory and name for a class database in a user-defined partition (processor set number 2 and higher). /usr/include/sys/class.h Header file defining the prototypes for routines in the libclass library, as well as the type, structure, and other kinds of definitions used by those routines. /usr/sbin/class_daemon Class scheduler daemon.

SEE ALSO

Commands: runclass(1), class_admin(8) Files: processor_sets(4) Functions: class_add(3), class_change(3), class_change_name(3), class_close(3), class_configure(3), class_create(3), class_database_file_exists(3), class_database_modified(3), class_database_name(3), class_delete(3), class_destroy(3), class_disable(3), class_get_class_members(3), class_get_classes(3), class_get_config_stats(3), class_load_database(3), class_open(3), class_restore_database(3), class_save_database(3)

Index Index for
Section 4
Index Alphabetical
listing for C
Top of page Top of
page