1*0f4c859eSApple OSS Distributions /* 2*0f4c859eSApple OSS Distributions * Copyright (c) 2017 Apple Inc. All rights reserved. 3*0f4c859eSApple OSS Distributions */ 4*0f4c859eSApple OSS Distributions 5*0f4c859eSApple OSS Distributions #pragma once 6*0f4c859eSApple OSS Distributions 7*0f4c859eSApple OSS Distributions #ifdef KERNEL_PRIVATE 8*0f4c859eSApple OSS Distributions #ifdef __cplusplus 9*0f4c859eSApple OSS Distributions 10*0f4c859eSApple OSS Distributions #include <IOKit/IOService.h> 11*0f4c859eSApple OSS Distributions #include <stdatomic.h> 12*0f4c859eSApple OSS Distributions #include <kern/bits.h> 13*0f4c859eSApple OSS Distributions #include <libkern/c++/OSPtr.h> 14*0f4c859eSApple OSS Distributions 15*0f4c859eSApple OSS Distributions struct thread_group; 16*0f4c859eSApple OSS Distributions 17*0f4c859eSApple OSS Distributions enum{ 18*0f4c859eSApple OSS Distributions kIOPerfControlClientWorkUntracked = 0, 19*0f4c859eSApple OSS Distributions }; 20*0f4c859eSApple OSS Distributions 21*0f4c859eSApple OSS Distributions /*! 22*0f4c859eSApple OSS Distributions * @class IOPerfControlClient : public OSObject 23*0f4c859eSApple OSS Distributions * @abstract Class which implements an interface allowing device drivers to participate in performance control. 24*0f4c859eSApple OSS Distributions * @discussion TODO 25*0f4c859eSApple OSS Distributions */ 26*0f4c859eSApple OSS Distributions class IOPerfControlClient final : public OSObject 27*0f4c859eSApple OSS Distributions { 28*0f4c859eSApple OSS Distributions OSDeclareDefaultStructors(IOPerfControlClient); 29*0f4c859eSApple OSS Distributions 30*0f4c859eSApple OSS Distributions protected: 31*0f4c859eSApple OSS Distributions virtual bool init(IOService *driver, uint64_t maxWorkCapacity); 32*0f4c859eSApple OSS Distributions virtual void free() APPLE_KEXT_OVERRIDE; 33*0f4c859eSApple OSS Distributions 34*0f4c859eSApple OSS Distributions public: 35*0f4c859eSApple OSS Distributions /*! 36*0f4c859eSApple OSS Distributions * @function copyClient 37*0f4c859eSApple OSS Distributions * @abstract Return a retained reference to a client object, to be released by the driver. It may be 38*0f4c859eSApple OSS Distributions * shared with other drivers in the system. 39*0f4c859eSApple OSS Distributions * @param driver The device driver that will be using this interface. 40*0f4c859eSApple OSS Distributions * @param maxWorkCapacity The maximum number of concurrent work items supported by the device driver. 41*0f4c859eSApple OSS Distributions * @returns An instance of IOPerfControlClient. 42*0f4c859eSApple OSS Distributions */ 43*0f4c859eSApple OSS Distributions static IOPerfControlClient *copyClient(IOService *driver, uint64_t maxWorkCapacity); 44*0f4c859eSApple OSS Distributions 45*0f4c859eSApple OSS Distributions /*! 46*0f4c859eSApple OSS Distributions * @function registerDevice 47*0f4c859eSApple OSS Distributions * @abstract Inform the system that work will be dispatched to a device in the future. 48*0f4c859eSApple OSS Distributions * @discussion The system will do some one-time setup work associated with the device, and may block the 49*0f4c859eSApple OSS Distributions * current thread during the setup. Devices should not be passed to work workSubmit, workSubmitAndBegin, 50*0f4c859eSApple OSS Distributions * workBegin, or workEnd until they have been successfully registered. The unregistration process happens 51*0f4c859eSApple OSS Distributions * automatically when the device object is deallocated. 52*0f4c859eSApple OSS Distributions * @param device The device object. Some platforms require device to be a specific subclass of IOService. 53*0f4c859eSApple OSS Distributions * @returns kIOReturnSuccess or an IOReturn error code 54*0f4c859eSApple OSS Distributions */ 55*0f4c859eSApple OSS Distributions virtual IOReturn registerDevice(IOService *driver, IOService *device); 56*0f4c859eSApple OSS Distributions 57*0f4c859eSApple OSS Distributions /*! 58*0f4c859eSApple OSS Distributions * @function unregisterDevice 59*0f4c859eSApple OSS Distributions * @abstract Inform the system that work will be no longer be dispatched to a device in the future. 60*0f4c859eSApple OSS Distributions * @discussion This call is optional as the unregistration process happens automatically when the device 61*0f4c859eSApple OSS Distributions * object is deallocated. This call may block the current thread and/or acquire locks. It should not be 62*0f4c859eSApple OSS Distributions * called until after all submitted work has been ended using workEnd. 63*0f4c859eSApple OSS Distributions * @param device The device object. Some platforms require device to be a specific subclass of IOService. 64*0f4c859eSApple OSS Distributions */ 65*0f4c859eSApple OSS Distributions virtual void unregisterDevice(IOService *driver, IOService *device); 66*0f4c859eSApple OSS Distributions 67*0f4c859eSApple OSS Distributions /*! 68*0f4c859eSApple OSS Distributions * @struct WorkSubmitArgs 69*0f4c859eSApple OSS Distributions * @discussion Drivers may submit additional device-specific arguments related to the submission of a work item 70*0f4c859eSApple OSS Distributions * by passing a struct with WorkSubmitArgs as its first member. Note: Drivers are responsible for publishing 71*0f4c859eSApple OSS Distributions * a header file describing these arguments. 72*0f4c859eSApple OSS Distributions */ 73*0f4c859eSApple OSS Distributions struct WorkSubmitArgs { 74*0f4c859eSApple OSS Distributions uint32_t version; 75*0f4c859eSApple OSS Distributions uint32_t size; 76*0f4c859eSApple OSS Distributions uint64_t submit_time; 77*0f4c859eSApple OSS Distributions uint64_t reserved[4]; 78*0f4c859eSApple OSS Distributions void *driver_data; 79*0f4c859eSApple OSS Distributions }; 80*0f4c859eSApple OSS Distributions 81*0f4c859eSApple OSS Distributions /*! 82*0f4c859eSApple OSS Distributions * @function workSubmit 83*0f4c859eSApple OSS Distributions * @abstract Tell the performance controller that work was submitted. 84*0f4c859eSApple OSS Distributions * @param device The device that will execute the work. Some platforms require device to be a 85*0f4c859eSApple OSS Distributions * specific subclass of IOService. 86*0f4c859eSApple OSS Distributions * @param args Optional device-specific arguments related to the submission of this work item. 87*0f4c859eSApple OSS Distributions * @returns A token representing this work item, which must be passed to workEnd when the work is finished 88*0f4c859eSApple OSS Distributions * unless the token equals kIOPerfControlClientWorkUntracked. Failure to do this will result in memory leaks 89*0f4c859eSApple OSS Distributions * and a degradation of system performance. 90*0f4c859eSApple OSS Distributions */ 91*0f4c859eSApple OSS Distributions virtual uint64_t workSubmit(IOService *device, WorkSubmitArgs *args = nullptr); 92*0f4c859eSApple OSS Distributions 93*0f4c859eSApple OSS Distributions /*! 94*0f4c859eSApple OSS Distributions * @struct WorkBeginArgs 95*0f4c859eSApple OSS Distributions * @discussion Drivers may submit additional device-specific arguments related to the start of a work item 96*0f4c859eSApple OSS Distributions * by passing a struct with WorkBeginArgs as its first member. Note: Drivers are responsible for publishing 97*0f4c859eSApple OSS Distributions * a header file describing these arguments. 98*0f4c859eSApple OSS Distributions */ 99*0f4c859eSApple OSS Distributions struct WorkBeginArgs { 100*0f4c859eSApple OSS Distributions uint32_t version; 101*0f4c859eSApple OSS Distributions uint32_t size; 102*0f4c859eSApple OSS Distributions uint64_t begin_time; 103*0f4c859eSApple OSS Distributions uint64_t reserved[4]; 104*0f4c859eSApple OSS Distributions void *driver_data; 105*0f4c859eSApple OSS Distributions }; 106*0f4c859eSApple OSS Distributions 107*0f4c859eSApple OSS Distributions /*! 108*0f4c859eSApple OSS Distributions * @function workSubmitAndBegin 109*0f4c859eSApple OSS Distributions * @abstract Tell the performance controller that work was submitted and immediately began executing. 110*0f4c859eSApple OSS Distributions * @param device The device that is executing the work. Some platforms require device to be a 111*0f4c859eSApple OSS Distributions * specific subclass of IOService. 112*0f4c859eSApple OSS Distributions * @param submitArgs Optional device-specific arguments related to the submission of this work item. 113*0f4c859eSApple OSS Distributions * @param beginArgs Optional device-specific arguments related to the start of this work item. 114*0f4c859eSApple OSS Distributions * @returns A token representing this work item, which must be passed to workEnd when the work is finished 115*0f4c859eSApple OSS Distributions * unless the token equals kIOPerfControlClientWorkUntracked. Failure to do this will result in memory leaks 116*0f4c859eSApple OSS Distributions * and a degradation of system performance. 117*0f4c859eSApple OSS Distributions */ 118*0f4c859eSApple OSS Distributions virtual uint64_t workSubmitAndBegin(IOService *device, WorkSubmitArgs *submitArgs = nullptr, 119*0f4c859eSApple OSS Distributions WorkBeginArgs *beginArgs = nullptr); 120*0f4c859eSApple OSS Distributions 121*0f4c859eSApple OSS Distributions /*! 122*0f4c859eSApple OSS Distributions * @function workBegin 123*0f4c859eSApple OSS Distributions * @abstract Tell the performance controller that previously submitted work began executing. 124*0f4c859eSApple OSS Distributions * @param device The device that is executing the work. Some platforms require device to be a 125*0f4c859eSApple OSS Distributions * specific subclass of IOService. 126*0f4c859eSApple OSS Distributions * @param args Optional device-specific arguments related to the start of this work item. 127*0f4c859eSApple OSS Distributions */ 128*0f4c859eSApple OSS Distributions virtual void workBegin(IOService *device, uint64_t token, WorkBeginArgs *args = nullptr); 129*0f4c859eSApple OSS Distributions 130*0f4c859eSApple OSS Distributions /*! 131*0f4c859eSApple OSS Distributions * @struct WorkEndArgs 132*0f4c859eSApple OSS Distributions * @discussion Drivers may submit additional device-specific arguments related to the end of a work item 133*0f4c859eSApple OSS Distributions * by passing a struct with WorkEndArgs as its first member. Note: Drivers are responsible for publishing 134*0f4c859eSApple OSS Distributions * a header file describing these arguments. 135*0f4c859eSApple OSS Distributions */ 136*0f4c859eSApple OSS Distributions struct WorkEndArgs { 137*0f4c859eSApple OSS Distributions uint32_t version; 138*0f4c859eSApple OSS Distributions uint32_t size; 139*0f4c859eSApple OSS Distributions uint64_t end_time; 140*0f4c859eSApple OSS Distributions uint64_t reserved[4]; 141*0f4c859eSApple OSS Distributions void *driver_data; 142*0f4c859eSApple OSS Distributions }; 143*0f4c859eSApple OSS Distributions 144*0f4c859eSApple OSS Distributions /*! 145*0f4c859eSApple OSS Distributions * @function workEnd 146*0f4c859eSApple OSS Distributions * @abstract Tell the performance controller that previously started work finished executing. 147*0f4c859eSApple OSS Distributions * @param device The device that executed the work. Some platforms require device to be a 148*0f4c859eSApple OSS Distributions * specific subclass of IOService. 149*0f4c859eSApple OSS Distributions * @param args Optional device-specific arguments related to the end of this work item. 150*0f4c859eSApple OSS Distributions * @param done Optional Set to false if the work has not yet completed. Drivers are then responsible for 151*0f4c859eSApple OSS Distributions * calling workBegin when the work resumes and workEnd with done set to True when it has completed. A workEnd() call 152*0f4c859eSApple OSS Distributions * without a corresponding workBegin() call is a way to cancel a work item and return token to IOPerfControl. 153*0f4c859eSApple OSS Distributions */ 154*0f4c859eSApple OSS Distributions virtual void workEnd(IOService *device, uint64_t token, WorkEndArgs *args = nullptr, bool done = true); 155*0f4c859eSApple OSS Distributions 156*0f4c859eSApple OSS Distributions /*! 157*0f4c859eSApple OSS Distributions * @function copyWorkContext 158*0f4c859eSApple OSS Distributions * @abstract Return a retained reference to an opaque OSObject, to be released by the driver. This object can 159*0f4c859eSApple OSS Distributions * be used by IOPerfControl to track a work item. This may perform dynamic memory allocation. 160*0f4c859eSApple OSS Distributions * @returns A pointer to an OSObject 161*0f4c859eSApple OSS Distributions */ 162*0f4c859eSApple OSS Distributions OSPtr<OSObject> copyWorkContext(); 163*0f4c859eSApple OSS Distributions 164*0f4c859eSApple OSS Distributions /*! 165*0f4c859eSApple OSS Distributions * @function workSubmitAndBeginWithContext 166*0f4c859eSApple OSS Distributions * @abstract Tell the performance controller that work was submitted and immediately began executing 167*0f4c859eSApple OSS Distributions * @param device The device that is executing the work. Some platforms require device to be a 168*0f4c859eSApple OSS Distributions * specific subclass of IOService. 169*0f4c859eSApple OSS Distributions * @param context An OSObject returned by copyWorkContext(). The context object will be used by IOPerfControl to track 170*0f4c859eSApple OSS Distributions * this work item. 171*0f4c859eSApple OSS Distributions * @param submitArgs Optional device-specific arguments related to the submission of this work item. 172*0f4c859eSApple OSS Distributions * @param beginArgs Optional device-specific arguments related to the start of this work item. 173*0f4c859eSApple OSS Distributions * @returns true if IOPerfControl is tracking this work item, else false. 174*0f4c859eSApple OSS Distributions * @note The workEndWithContext() call is optional if the corresponding workSubmitWithContext() call returned false. 175*0f4c859eSApple OSS Distributions */ 176*0f4c859eSApple OSS Distributions bool workSubmitAndBeginWithContext(IOService *device, OSObject *context, WorkSubmitArgs *submitArgs = nullptr, 177*0f4c859eSApple OSS Distributions WorkBeginArgs *beginArgs = nullptr); 178*0f4c859eSApple OSS Distributions 179*0f4c859eSApple OSS Distributions /*! 180*0f4c859eSApple OSS Distributions * @function workSubmitWithContext 181*0f4c859eSApple OSS Distributions * @abstract Tell the performance controller that work was submitted. 182*0f4c859eSApple OSS Distributions * @param device The device that will execute the work. Some platforms require device to be a 183*0f4c859eSApple OSS Distributions * specific subclass of IOService. 184*0f4c859eSApple OSS Distributions * @param context An OSObject returned by copyWorkContext(). The context object will be used by IOPerfControl to track 185*0f4c859eSApple OSS Distributions * this work item. 186*0f4c859eSApple OSS Distributions * @param args Optional device-specific arguments related to the submission of this work item. 187*0f4c859eSApple OSS Distributions * @returns true if IOPerfControl is tracking this work item, else false. 188*0f4c859eSApple OSS Distributions */ 189*0f4c859eSApple OSS Distributions bool workSubmitWithContext(IOService *device, OSObject *context, WorkSubmitArgs *args = nullptr); 190*0f4c859eSApple OSS Distributions 191*0f4c859eSApple OSS Distributions /*! 192*0f4c859eSApple OSS Distributions * @function workBeginWithContext 193*0f4c859eSApple OSS Distributions * @abstract Tell the performance controller that previously submitted work began executing. 194*0f4c859eSApple OSS Distributions * @param device The device that is executing the work. Some platforms require device to be a 195*0f4c859eSApple OSS Distributions * specific subclass of IOService. 196*0f4c859eSApple OSS Distributions * @param context An OSObject returned by copyWorkContext() and provided to the previous call to workSubmitWithContext(). 197*0f4c859eSApple OSS Distributions * @param args Optional device-specific arguments related to the start of this work item. 198*0f4c859eSApple OSS Distributions * @note The workBeginWithContext() and workEndWithContext() calls are optional if the corresponding workSubmitWithContext() call returned false. 199*0f4c859eSApple OSS Distributions */ 200*0f4c859eSApple OSS Distributions void workBeginWithContext(IOService *device, OSObject *context, WorkBeginArgs *args = nullptr); 201*0f4c859eSApple OSS Distributions 202*0f4c859eSApple OSS Distributions /*! 203*0f4c859eSApple OSS Distributions * @function workEndWithContext 204*0f4c859eSApple OSS Distributions * @abstract Tell the performance controller that previously started work finished executing. 205*0f4c859eSApple OSS Distributions * @param device The device that executed the work. Some platforms require device to be a 206*0f4c859eSApple OSS Distributions * specific subclass of IOService. 207*0f4c859eSApple OSS Distributions * @param context An OSObject returned by copyWorkContext() and provided to the previous call to workSubmitWithContext(). 208*0f4c859eSApple OSS Distributions * @param args Optional device-specific arguments related to the end of this work item. 209*0f4c859eSApple OSS Distributions * @param done Optional Set to false if the work has not yet completed. Drivers are then responsible for 210*0f4c859eSApple OSS Distributions * calling workBegin when the work resumes and workEnd with done set to True when it has completed. 211*0f4c859eSApple OSS Distributions * @note The workEndWithContext() call is optional if the corresponding workSubmitWithContext() call returned false. A workEndWithContext() 212*0f4c859eSApple OSS Distributions * call without a corresponding workBeginWithContext() call is a way to cancel a work item. 213*0f4c859eSApple OSS Distributions */ 214*0f4c859eSApple OSS Distributions void workEndWithContext(IOService *device, OSObject *context, WorkEndArgs *args = nullptr, bool done = true); 215*0f4c859eSApple OSS Distributions 216*0f4c859eSApple OSS Distributions /*! 217*0f4c859eSApple OSS Distributions * @struct WorkUpdateArgs 218*0f4c859eSApple OSS Distributions * @discussion Drivers may submit additional device-specific arguments related to a work item by passing a 219*0f4c859eSApple OSS Distributions * struct with WorkUpdateArgs as its first member. Note: Drivers are responsible for publishing 220*0f4c859eSApple OSS Distributions * a header file describing these arguments. 221*0f4c859eSApple OSS Distributions */ 222*0f4c859eSApple OSS Distributions struct WorkUpdateArgs { 223*0f4c859eSApple OSS Distributions uint32_t version; 224*0f4c859eSApple OSS Distributions uint32_t size; 225*0f4c859eSApple OSS Distributions uint64_t update_time; 226*0f4c859eSApple OSS Distributions uint64_t reserved[4]; 227*0f4c859eSApple OSS Distributions void *driver_data; 228*0f4c859eSApple OSS Distributions }; 229*0f4c859eSApple OSS Distributions 230*0f4c859eSApple OSS Distributions /*! 231*0f4c859eSApple OSS Distributions * @function workUpdateWithContext 232*0f4c859eSApple OSS Distributions * @abstract Provide and receive additional information from the performance controller. If this call is 233*0f4c859eSApple OSS Distributions * made at all, it should be between workSubmit and workEnd. The purpose and implementation of this call are 234*0f4c859eSApple OSS Distributions * device specific, and may do nothing on some devices. 235*0f4c859eSApple OSS Distributions * @param device The device that submitted the work. Some platforms require device to be a 236*0f4c859eSApple OSS Distributions * specific subclass of IOService. 237*0f4c859eSApple OSS Distributions * @param context An OSObject returned by copyWorkContext() and provided to the previous call to workSubmitWithContext(). 238*0f4c859eSApple OSS Distributions * @param args Optional device-specific arguments. 239*0f4c859eSApple OSS Distributions */ 240*0f4c859eSApple OSS Distributions void workUpdateWithContext(IOService *device, OSObject *context, WorkUpdateArgs *args = nullptr); 241*0f4c859eSApple OSS Distributions 242*0f4c859eSApple OSS Distributions /* 243*0f4c859eSApple OSS Distributions * Callers should always use the CURRENT version so that the kernel can detect both older 244*0f4c859eSApple OSS Distributions * and newer structure layouts. New callbacks should always be added at the end of the 245*0f4c859eSApple OSS Distributions * structure, and xnu should expect existing source recompiled against newer headers 246*0f4c859eSApple OSS Distributions * to pass NULL for unimplemented callbacks. 247*0f4c859eSApple OSS Distributions */ 248*0f4c859eSApple OSS Distributions 249*0f4c859eSApple OSS Distributions #define PERFCONTROL_INTERFACE_VERSION_NONE (0) /* no interface */ 250*0f4c859eSApple OSS Distributions #define PERFCONTROL_INTERFACE_VERSION_1 (1) /* up-to workEnd */ 251*0f4c859eSApple OSS Distributions #define PERFCONTROL_INTERFACE_VERSION_2 (2) /* up-to workUpdate */ 252*0f4c859eSApple OSS Distributions #define PERFCONTROL_INTERFACE_VERSION_3 (3) /* up-to (un)registerDriverDevice */ 253*0f4c859eSApple OSS Distributions #define PERFCONTROL_INTERFACE_VERSION_CURRENT PERFCONTROL_INTERFACE_VERSION_3 254*0f4c859eSApple OSS Distributions 255*0f4c859eSApple OSS Distributions /*! 256*0f4c859eSApple OSS Distributions * @struct PerfControllerInterface 257*0f4c859eSApple OSS Distributions * @discussion Function pointers necessary to register a performance controller. Not for general driver use. 258*0f4c859eSApple OSS Distributions */ 259*0f4c859eSApple OSS Distributions struct PerfControllerInterface { 260*0f4c859eSApple OSS Distributions struct DriverState { 261*0f4c859eSApple OSS Distributions uint32_t has_target_thread_group : 1; 262*0f4c859eSApple OSS Distributions uint32_t has_device_info : 1; 263*0f4c859eSApple OSS Distributions uint32_t reserved : 30; 264*0f4c859eSApple OSS Distributions 265*0f4c859eSApple OSS Distributions uint64_t target_thread_group_id; 266*0f4c859eSApple OSS Distributions void *target_thread_group_data; 267*0f4c859eSApple OSS Distributions 268*0f4c859eSApple OSS Distributions uint32_t device_type; 269*0f4c859eSApple OSS Distributions uint32_t instance_id; 270*0f4c859eSApple OSS Distributions }; 271*0f4c859eSApple OSS Distributions 272*0f4c859eSApple OSS Distributions struct WorkState { 273*0f4c859eSApple OSS Distributions uint64_t thread_group_id; 274*0f4c859eSApple OSS Distributions void *thread_group_data; 275*0f4c859eSApple OSS Distributions void *work_data; 276*0f4c859eSApple OSS Distributions uint32_t work_data_size; 277*0f4c859eSApple OSS Distributions uint32_t started : 1; 278*0f4c859eSApple OSS Distributions uint32_t reserved : 31; 279*0f4c859eSApple OSS Distributions const DriverState* driver_state; 280*0f4c859eSApple OSS Distributions }; 281*0f4c859eSApple OSS Distributions 282*0f4c859eSApple OSS Distributions using RegisterDeviceFunction = IOReturn (*)(IOService *); 283*0f4c859eSApple OSS Distributions using RegisterDriverDeviceFunction = IOReturn (*)(IOService *, IOService *, DriverState *); 284*0f4c859eSApple OSS Distributions using WorkCanSubmitFunction = bool (*)(IOService *, WorkState *, WorkSubmitArgs *); 285*0f4c859eSApple OSS Distributions using WorkSubmitFunction = void (*)(IOService *, uint64_t, WorkState *, WorkSubmitArgs *); 286*0f4c859eSApple OSS Distributions using WorkBeginFunction = void (*)(IOService *, uint64_t, WorkState *, WorkBeginArgs *); 287*0f4c859eSApple OSS Distributions using WorkEndFunction = void (*)(IOService *, uint64_t, WorkState *, WorkEndArgs *, bool); 288*0f4c859eSApple OSS Distributions using WorkUpdateFunction = void (*)(IOService *, uint64_t, WorkState *, WorkUpdateArgs *); 289*0f4c859eSApple OSS Distributions 290*0f4c859eSApple OSS Distributions uint64_t version; 291*0f4c859eSApple OSS Distributions RegisterDeviceFunction registerDevice; 292*0f4c859eSApple OSS Distributions RegisterDeviceFunction unregisterDevice; 293*0f4c859eSApple OSS Distributions WorkCanSubmitFunction workCanSubmit; 294*0f4c859eSApple OSS Distributions WorkSubmitFunction workSubmit; 295*0f4c859eSApple OSS Distributions WorkBeginFunction workBegin; 296*0f4c859eSApple OSS Distributions WorkEndFunction workEnd; 297*0f4c859eSApple OSS Distributions WorkUpdateFunction workUpdate; 298*0f4c859eSApple OSS Distributions RegisterDriverDeviceFunction registerDriverDevice; 299*0f4c859eSApple OSS Distributions RegisterDriverDeviceFunction unregisterDriverDevice; 300*0f4c859eSApple OSS Distributions }; 301*0f4c859eSApple OSS Distributions 302*0f4c859eSApple OSS Distributions struct IOPerfControlClientShared { 303*0f4c859eSApple OSS Distributions atomic_uint_fast8_t maxDriverIndex; 304*0f4c859eSApple OSS Distributions PerfControllerInterface interface; 305*0f4c859eSApple OSS Distributions IOLock *interfaceLock; 306*0f4c859eSApple OSS Distributions OSSet *deviceRegistrationList; 307*0f4c859eSApple OSS Distributions }; 308*0f4c859eSApple OSS Distributions 309*0f4c859eSApple OSS Distributions struct IOPerfControlClientData { 310*0f4c859eSApple OSS Distributions struct thread_group *target_thread_group; 311*0f4c859eSApple OSS Distributions PerfControllerInterface::DriverState driverState; 312*0f4c859eSApple OSS Distributions IOService* device; 313*0f4c859eSApple OSS Distributions }; 314*0f4c859eSApple OSS Distributions /*! 315*0f4c859eSApple OSS Distributions * @function registerPerformanceController 316*0f4c859eSApple OSS Distributions * @abstract Register a performance controller to receive callbacks. Not for general driver use. 317*0f4c859eSApple OSS Distributions * @param interface Struct containing callback functions implemented by the performance controller. 318*0f4c859eSApple OSS Distributions * @returns kIOReturnSuccess or kIOReturnError if the interface was already registered. 319*0f4c859eSApple OSS Distributions */ 320*0f4c859eSApple OSS Distributions virtual IOReturn registerPerformanceController(PerfControllerInterface *interface); 321*0f4c859eSApple OSS Distributions 322*0f4c859eSApple OSS Distributions /*! 323*0f4c859eSApple OSS Distributions * @function getClientData 324*0f4c859eSApple OSS Distributions * @abstract Not for general driver use. Only used by registerPerformanceController(). Allows performanceController to register existing IOPerfControlClient. 325*0f4c859eSApple OSS Distributions * @returns IOPerfControlData associated with a IOPerfControlClient 326*0f4c859eSApple OSS Distributions */ 327*0f4c859eSApple OSS Distributions IOPerfControlClientData * getClientData()328*0f4c859eSApple OSS Distributions getClientData() 329*0f4c859eSApple OSS Distributions { 330*0f4c859eSApple OSS Distributions return &clientData; 331*0f4c859eSApple OSS Distributions } 332*0f4c859eSApple OSS Distributions 333*0f4c859eSApple OSS Distributions private: 334*0f4c859eSApple OSS Distributions struct WorkTableEntry { 335*0f4c859eSApple OSS Distributions struct thread_group *thread_group; 336*0f4c859eSApple OSS Distributions bool started; 337*0f4c859eSApple OSS Distributions uint8_t perfcontrol_data[32]; 338*0f4c859eSApple OSS Distributions }; 339*0f4c859eSApple OSS Distributions 340*0f4c859eSApple OSS Distributions static constexpr size_t kMaxWorkTableNumEntries = 1024; 341*0f4c859eSApple OSS Distributions static constexpr size_t kWorkTableIndexBits = 24; 342*0f4c859eSApple OSS Distributions static constexpr size_t kWorkTableMaxSize = (1 << kWorkTableIndexBits) - 1; // - 1 since 343*0f4c859eSApple OSS Distributions // kIOPerfControlClientWorkUntracked takes number 0 344*0f4c859eSApple OSS Distributions static constexpr size_t kWorkTableIndexMask = (const size_t)mask(kWorkTableIndexBits); 345*0f4c859eSApple OSS Distributions 346*0f4c859eSApple OSS Distributions uint64_t allocateToken(thread_group *thread_group); 347*0f4c859eSApple OSS Distributions void deallocateToken(uint64_t token); 348*0f4c859eSApple OSS Distributions WorkTableEntry *getEntryForToken(uint64_t token); 349*0f4c859eSApple OSS Distributions void markEntryStarted(uint64_t token, bool started); 350*0f4c859eSApple OSS Distributions inline uint64_t tokenToGlobalUniqueToken(uint64_t token); 351*0f4c859eSApple OSS Distributions 352*0f4c859eSApple OSS Distributions uint8_t driverIndex; 353*0f4c859eSApple OSS Distributions IOPerfControlClientShared *shared; 354*0f4c859eSApple OSS Distributions WorkTableEntry *workTable; 355*0f4c859eSApple OSS Distributions size_t workTableLength; 356*0f4c859eSApple OSS Distributions size_t workTableNextIndex; 357*0f4c859eSApple OSS Distributions IOSimpleLock *workTableLock; 358*0f4c859eSApple OSS Distributions 359*0f4c859eSApple OSS Distributions IOPerfControlClientData clientData; 360*0f4c859eSApple OSS Distributions }; 361*0f4c859eSApple OSS Distributions 362*0f4c859eSApple OSS Distributions #endif /* __cplusplus */ 363*0f4c859eSApple OSS Distributions #endif /* KERNEL_PRIVATE */ 364