|
Name |
|
Date |
Size |
#Lines |
LOC |
| .. | | - | - |
| .gitignore | D | 03-May-2024 | 23 | 3 | 2 |
| README | D | 03-May-2024 | 3.2 KiB | 61 | 52 |
| gen4.xml | D | 03-May-2024 | 2.6 KiB | 53 | 52 |
| gen45.xml | D | 03-May-2024 | 2.8 KiB | 57 | 56 |
| gen5.xml | D | 03-May-2024 | 2.8 KiB | 57 | 56 |
| gen6.xml | D | 03-May-2024 | 110.9 KiB | 1,973 | 1,872 |
| gen7.xml | D | 03-May-2024 | 142.3 KiB | 2,549 | 2,410 |
| gen75.xml | D | 03-May-2024 | 166.5 KiB | 2,965 | 2,794 |
| gen8.xml | D | 03-May-2024 | 180.9 KiB | 3,204 | 3,024 |
| gen9.xml | D | 03-May-2024 | 196 KiB | 3,475 | 3,290 |
| genX_pack.h | D | 03-May-2024 | 1.8 KiB | 52 | 25 |
| gen_macros.h | D | 03-May-2024 | 2.9 KiB | 93 | 36 |
| gen_pack_header.py | D | 03-May-2024 | 21.2 KiB | 658 | 532 |
README
1This provides some background the design of the generated headers. We
2started out trying to generate bit fields but it evolved into the pack
3functions because of a few limitations:
4
5 1) Bit fields still generate terrible code today. Even with modern
6 optimizing compilers you get multiple load+mask+store operations
7 to the same dword in memory as you set individual bits. The
8 compiler also has to generate code to mask out overflowing values
9 (for example, if you assign 200 to a 2 bit field). Our driver
10 never writes overflowing values so that's not needed. On the
11 other hand, most compiler recognize that the template struct we
12 use is a temporary variable and copy propagate the individual
13 fields and do amazing constant folding. You should take a look
14 at the code that gets generated when you compile in release mode
15 with optimizations.
16
17 2) For some types we need to have overlapping bit fields. For
18 example, some values are 64 byte aligned 32 bit offsets. The
19 lower 5 bits of the offset are always zero, so the hw packs in a
20 few misc bits in the lower 5 bits there. Other times a field can
21 be either a u32 or a float. I tried to do this with overlapping
22 anonymous unions and it became a big mess. Also, when using
23 initializers, you can only initialize one union member so this
24 just doesn't work with out approach.
25
26 The pack functions on the other hand allows us a great deal of
27 flexibility in how we combine things. In the case of overlapping
28 fields (the u32 and float case), if we only set one of them in
29 the pack function, the compiler will recognize that the other is
30 initialized to 0 and optimize out the code to or it it.
31
32 3) Bit fields (and certainly overlapping anonymous unions of bit
33 fields) aren't generally stable across compilers in how they're
34 laid out and aligned. Our pack functions let us control exactly
35 how things get packed, using only simple and unambiguous bitwise
36 shifting and or'ing that works on any compiler.
37
38Once we have the pack function it allows us to hook in various
39transformations and validation as we go from template struct to dwords
40in memory:
41
42 1) Validation: As I said above, our driver isn't supposed to write
43 overflowing values to the fields, but we've of course had lots of
44 cases where we make mistakes and write overflowing values. With
45 the pack function, we can actually assert on that and catch it at
46 runtime. bitfields would just silently truncate.
47
48 2) Type conversions: some times it's just a matter of writing a
49 float to a u32, but we also convert from bool to bits, from
50 floats to fixed point integers.
51
52 3) Relocations: whenever we have a pointer from one buffer to
53 another (for example a pointer from the meta data for a texture
54 to the raw texture data), we have to tell the kernel about it so
55 it can adjust the pointer to point to the final location. That
56 means extra work we have to do extra work to record and annotate
57 the dword location that holds the pointer. With bit fields, we'd
58 have to call a function to do this, but with the pack function we
59 generate code in the pack function to do this for us. That's a
60 lot less error prone and less work.
61