1*0f4c859eSApple OSS DistributionsKernel Data Descriptors 2*0f4c859eSApple OSS Distributions======================= 3*0f4c859eSApple OSS Distributions 4*0f4c859eSApple OSS DistributionsThis project allows for dynamic data to be passed from the kernel to userspace tools without binding them to particular version of 5*0f4c859eSApple OSS Distributionsstruct definition. The `libkdd` library provides convenient API for parsing and interpreting `kernel chunked data`. 6*0f4c859eSApple OSS Distributions 7*0f4c859eSApple OSS DistributionsThe libkdd APIs are defined in [kdd.h](./kdd.h) 8*0f4c859eSApple OSS Distributions 9*0f4c859eSApple OSS DistributionsThe `KCDATA` format 10*0f4c859eSApple OSS Distributions=================== 11*0f4c859eSApple OSS Distributions 12*0f4c859eSApple OSS DistributionsThe format for data is setup in a generic format as follows 13*0f4c859eSApple OSS Distributions 14*0f4c859eSApple OSS DistributionsLayout of data structure 15*0f4c859eSApple OSS Distributions------------------------ 16*0f4c859eSApple OSS Distributions 17*0f4c859eSApple OSS Distributions | 8 - bytes | 18*0f4c859eSApple OSS Distributions |---------------------------| ------ offset = 00 19*0f4c859eSApple OSS Distributions | type = MAGIC | LENGTH | # BEGIN Header 20*0f4c859eSApple OSS Distributions | 0 | 21*0f4c859eSApple OSS Distributions |---------------------------| ------ offset = 16 22*0f4c859eSApple OSS Distributions | type | size | # chunk header 23*0f4c859eSApple OSS Distributions | flags | 24*0f4c859eSApple OSS Distributions |---------------------------| ------ offset = 32 25*0f4c859eSApple OSS Distributions | data | # arbitrary data (len=16) 26*0f4c859eSApple OSS Distributions |___________data____________| 27*0f4c859eSApple OSS Distributions |---------------------------| ------ offset = 48 28*0f4c859eSApple OSS Distributions | type | size | # chunk header 29*0f4c859eSApple OSS Distributions | flags | 30*0f4c859eSApple OSS Distributions |---------------------------| ------ offset = 64 31*0f4c859eSApple OSS Distributions | data | # arbitrary data (len=32) 32*0f4c859eSApple OSS Distributions | data | 33*0f4c859eSApple OSS Distributions | data | 34*0f4c859eSApple OSS Distributions |___________data____________| 35*0f4c859eSApple OSS Distributions |---------------------------| ------ offset = 96 36*0f4c859eSApple OSS Distributions | type = END | size=0 | # chunk header 37*0f4c859eSApple OSS Distributions | 0 | 38*0f4c859eSApple OSS Distributions 39*0f4c859eSApple OSS Distributions 40*0f4c859eSApple OSS DistributionsThe type field describes what kind of data is passed. For example type = `TASK_CRASHINFO_UUID` means the following data is a uuid. 41*0f4c859eSApple OSS DistributionsThese types need to be defined in task_corpses.h for easy consumption by userspace inspection tools. 42*0f4c859eSApple OSS Distributions 43*0f4c859eSApple OSS DistributionsSome range of types is reserved for special types like ints, longs etc. A cool new functionality made possible with this 44*0f4c859eSApple OSS Distributionsextensible data format is that kernel can decide to put more information as required without requiring user space tools to 45*0f4c859eSApple OSS Distributionsre-compile to be compatible. The case of `rusage` struct versions could be introduced without breaking existing tools. 46*0f4c859eSApple OSS Distributions 47*0f4c859eSApple OSS DistributionsFeature description: Generic data with description 48*0f4c859eSApple OSS Distributions------------------- 49*0f4c859eSApple OSS DistributionsFurther more generic data with description is very much possible now. For example 50*0f4c859eSApple OSS Distributions 51*0f4c859eSApple OSS Distributions - kcdata_add_uint64_with_description(cdatainfo, 0x700, "NUM MACH PORTS"); 52*0f4c859eSApple OSS Distributions - and more functions that allow adding description. 53*0f4c859eSApple OSS Distributions 54*0f4c859eSApple OSS DistributionsThe userspace tools can then look at the description and print the data even if they are not compiled with knowledge of the field apriori. 55*0f4c859eSApple OSS Distributions 56*0f4c859eSApple OSS Distributions Example data: 57*0f4c859eSApple OSS Distributions 0000 57 f1 ad de 00 00 00 00 00 00 00 00 00 00 00 00 W............... 58*0f4c859eSApple OSS Distributions 0010 01 00 00 00 00 00 00 00 30 00 00 00 00 00 00 00 ........0....... 59*0f4c859eSApple OSS Distributions 0020 50 49 44 00 00 00 00 00 00 00 00 00 00 00 00 00 PID............. 60*0f4c859eSApple OSS Distributions 0030 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 ................ 61*0f4c859eSApple OSS Distributions 0040 9c 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 ................ 62*0f4c859eSApple OSS Distributions 0050 01 00 00 00 00 00 00 00 30 00 00 00 00 00 00 00 ........0....... 63*0f4c859eSApple OSS Distributions 0060 50 41 52 45 4e 54 20 50 49 44 00 00 00 00 00 00 PARENT PID...... 64*0f4c859eSApple OSS Distributions 0070 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 ................ 65*0f4c859eSApple OSS Distributions 0080 01 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 ................ 66*0f4c859eSApple OSS Distributions 0090 ed 58 91 f1 67*0f4c859eSApple OSS Distributions 68*0f4c859eSApple OSS Distributions 69*0f4c859eSApple OSS DistributionsFeature description: Container markers for compound data 70*0f4c859eSApple OSS Distributions------------------ 71*0f4c859eSApple OSS Distributions 72*0f4c859eSApple OSS DistributionsIf a given kernel data type is complex and requires adding multiple optional fields inside a container 73*0f4c859eSApple OSS Distributionsobject for a consumer to understand arbitrary data, we package it using container markers. 74*0f4c859eSApple OSS Distributions 75*0f4c859eSApple OSS DistributionsFor example, the stackshot code gathers information and describes the state of a given task with respect 76*0f4c859eSApple OSS Distributionsto many subsystems. It includes data such as io stats, vm counters, process names/flags and syscall counts. 77*0f4c859eSApple OSS Distributions 78*0f4c859eSApple OSS Distributions kcdata_add_container_marker(kcdata_p, KCDATA_TYPE_CONTAINER_BEGIN, STACKSHOT_KCCONTAINER_TASK, task_uniqueid); 79*0f4c859eSApple OSS Distributions // add multiple data, or add_<type>_with_description()s here 80*0f4c859eSApple OSS Distributions 81*0f4c859eSApple OSS Distributions kcdata_add_container_marker(kcdata_p, KCDATA_TYPE_CONTAINER_END, STACKSHOT_KCCONTAINER_TASK, task_uniqueid); 82*0f4c859eSApple OSS Distributions 83*0f4c859eSApple OSS Distributions 84*0f4c859eSApple OSS DistributionsFeature description: Custom Data formats on demand 85*0f4c859eSApple OSS Distributions-------------------- 86*0f4c859eSApple OSS Distributions 87*0f4c859eSApple OSS DistributionsWith the self describing nature of format, the kernel provider can describe a data type (uniquely identified by a number) and use 88*0f4c859eSApple OSS Distributionsit in the buffer for sending data. The consumer can parse the type information and have knowledge of describing incoming data. 89*0f4c859eSApple OSS DistributionsFollowing is an example of how we can describe a kernel specific struct sample_disk_io_stats in buffer. 90*0f4c859eSApple OSS Distributions 91*0f4c859eSApple OSS Distributions struct sample_disk_io_stats { 92*0f4c859eSApple OSS Distributions uint64_t disk_reads_count; 93*0f4c859eSApple OSS Distributions uint64_t disk_reads_size; 94*0f4c859eSApple OSS Distributions uint64_t io_priority_count[4]; 95*0f4c859eSApple OSS Distributions uint64_t io_priority_size; 96*0f4c859eSApple OSS Distributions } __attribute__ ((packed)); 97*0f4c859eSApple OSS Distributions 98*0f4c859eSApple OSS Distributions 99*0f4c859eSApple OSS Distributions struct kcdata_subtype_descriptor disk_io_stats_def[] = { 100*0f4c859eSApple OSS Distributions {KCS_SUBTYPE_FLAGS_NONE, KC_ST_UINT64, 0 * sizeof(uint64_t), sizeof(uint64_t), "disk_reads_count"}, 101*0f4c859eSApple OSS Distributions {KCS_SUBTYPE_FLAGS_NONE, KC_ST_UINT64, 1 * sizeof(uint64_t), sizeof(uint64_t), "disk_reads_size"}, 102*0f4c859eSApple OSS Distributions {KCS_SUBTYPE_FLAGS_ARRAY, KC_ST_UINT64, 2 * sizeof(uint64_t), KCS_SUBTYPE_PACK_SIZE(4, sizeof(uint64_t)), "io_priority_count"}, 103*0f4c859eSApple OSS Distributions {KCS_SUBTYPE_FLAGS_ARRAY, KC_ST_UINT64, (2 + 4) * sizeof(uint64_t), sizeof(uint64_t), "io_priority_size"}, 104*0f4c859eSApple OSS Distributions }; 105*0f4c859eSApple OSS Distributions 106*0f4c859eSApple OSS DistributionsNow you can add this custom type definition into the buffer as 107*0f4c859eSApple OSS Distributions kcdata_add_type_definition(kcdata_p, KCTYPE_SAMPLE_DISK_IO_STATS, "sample_disk_io_stats", 108*0f4c859eSApple OSS Distributions &disk_io_stats_def[0], sizeof(disk_io_stats_def)/sizeof(struct kcdata_subtype_descriptor)); 109*0f4c859eSApple OSS Distributions 110