Den 11.03.2021 15.48, skrev Peter Stuge: > Noralf Trønnes wrote: >>> I didn't receive the expected bits/bytes for RGB111 on the bulk endpoint, >>> I think because of how components were extracted in gud_xrgb8888_to_color(). >>> >>> Changing to the following gets me the expected (X R1 G1 B1 X R2 G2 B2) bytes: >>> >>> r = (*pix32 >> 8) & 0xff; >>> g = (*pix32 >> 16) & 0xff; >>> b = (*pix32++ >> 24) & 0xff; >> >> We're accessing the whole word here through pix32, no byte access, so >> endianess doesn't come into play. > > Endianness matters because parts of pix32 are used. > This code: #include <stdio.h> #include <stdint.h> void main() { volatile uint32_t endian = 0x01234567; uint32_t v = 0xaabbccdd; uint32_t *pix32 = &v; uint8_t r, g, b, *p; r = *pix32 >> 16; g = *pix32 >> 8; b = *pix32++; printf("xrgb8888=%08x\n", v); printf("32-bit access:\n"); printf("r=%02x\n", r); printf("g=%02x\n", g); printf("b=%02x\n", b); printf("Byte access on %s:\n", (*((uint8_t*)(&endian))) == 0x67 ? "LE" : "BE"); p = (uint8_t *)&v; printf("r=%02x\n", p[1]); printf("g=%02x\n", p[2]); printf("b=%02x\n", p[3]); } prints: xrgb8888=aabbccdd 32-bit access: r=bb g=cc b=dd Byte access on LE: r=cc g=bb b=aa > Software only sees bytes (or larger) because addresses are byte granular, > but must pay attention to the bit order when dealing with smaller values > inside larger memory accesses. > > Given 4 bytes of memory {0x11, 0x22, 0x33, 0x44} at address A, both LE > and BE machines appear the same when accessing individual bytes, but with > uint32_t *a32 = A then a32[0] is 0x44332211 on LE and 0x11223344 on BE. > > > Hence the question: What does DRM promise about the XRGB8888 mode? > That it's a 32-bit value. From include/uapi/drm/drm_fourcc.h: /* 32 bpp RGB */ #define DRM_FORMAT_XRGB8888 fourcc_code('X', 'R', '2', '4') /* [31:0] x:R:G:B 8:8:8:8 little endian */ If a raw buffer was passed from a BE to an LE machine, there would be problems because of how the value is stored, but here it's the same endianness in userspace and kernel space. There is code in gud_prep_flush() that handles a BE host with a multibyte format: } else if (gud_is_big_endian() && format->cpp[0] > 1) { drm_fb_swab(buf, vaddr, fb, rect, !import_attach); In this case we can't just pass on the raw buffer to the device since the protocol is LE, and thus have to swap the bytes to match up how they're stored in memory on the device. I'm not loosing any of the colors when running modetest. This is the test image that modetest uses and it comes through just like that: https://commons.wikimedia.org/wiki/File:SMPTE_Color_Bars.svg Noralf. > Is it guaranteed that the first byte in memory is always unused, the second > represents red, the third green and the fourth blue (endianess agnostic)? > I'd expect this and I guess that it is the case, but I don't know DRM? > > Or is it instead guaranteed that when accessed natively as one 32-bit > value the blue component is always in the most significant byte (endianess > abstracted, always LE in memory) or always in the least significant byte > (endianess abstracted, always BE in memory)? > This would be annoying for userspace, but well, it's possible. > > In the abstracted (latter) case pix32 would work, but could still be > questioned on style, and in fact, pix32 didn't work for me, so at a > minimum the byte order would be the reverse. > > > In the agnostic (former) case your code was correct for BE and mine > for LE, but I'd then suggest using a u8 * to both work correctly > everywhere and be obvious. > > >> This change will flip r and b, which gives: XRGB8888 -> XBGR1111 > > The old code was: > r = *pix32 >> 16; > g = *pix32 >> 8; > b = *pix32++; > > On my LE machine this set r to the third byte (G), g to the second (R) > and b to the first (X), explaining the color confusion that I saw. > > >> BGR is a common thing on controllers, are you sure yours are set to RGB >> and not BGR? > > Yes; I've verified that my display takes red in MSB both in its data > sheet and by writing raw bits to it on a system without the gud driver. > > >> And the 0xff masking isn't necessary since we're assigning to a byte, right? > > Not strictly neccessary but I like to do it anyway, both to be explicit > and also to ensure that the compiler will never sign extend, if types > are changed or if values become treated as signed and/or larger by the > compiler because the code is changed. > > It's frustrating to debug such unexpected changes in behavior due to > a type change or calculation change, but if you find it too defensive > then go ahead and remove it, if pix32 does stay. > > >> I haven't got a native R1G1B1 display so I have emulated and I do get >> the expected colors. This is the conversion function I use on the device >> which I think is correct: >> >> static size_t rgb111_to_rgb565(uint16_t *dst, uint8_t *src, >> uint16_t src_width, uint16_t src_height) >> { >> uint8_t rgb111, val = 0; >> size_t len = 0; >> >> for (uint16_t y = 0; y < src_height; y++) { >> for (uint16_t x = 0; x < src_width; x++) { >> if (!(x % 2)) >> val = *src++; >> rgb111 = val >> 4; >> *dst++ = ((rgb111 & 0x04) << 13) | ((rgb111 & 0x02) << 9) | >> ((rgb111 & 0x01) << 4); > > I'm afraid this isn't correct. Two wrongs end up cancelling each other > out and it's not so obvious because the destination has symmetric (565) > components. > > If you were to convert to xrgb8888 in the same way I think you'd also > see some color confusion, and in any case blue is getting lost already > in gud_xrgb8888_to_color() on LE. > > > //Peter >