mirror of
https://github.com/mpv-player/mpv
synced 2025-01-25 00:53:22 +00:00
video: some concessions to big endian hosts
The recent changes to the image format metadata broke big endian, and that was intentional. Some things are inherent to little endian (like the idea to coalesce bit and byte offsets into a single bit offset), and they don't be fixed. But some obvious things can be fixed, such as marking LE vs. BE formats the right way around on BE hosts. The metadata is formally still in LE, except that if the LE/BE flag matches the host endian, the host endian can be used when accessing packed formats with bit shifts, or when computing byte aligned component byte offsets. The former may work because formats with LE/BE variants use the same bit offsets after byte swapping, the latter may work because little endian is the natural concept for addressing memory. But it will "subtly" fail to do the right thing in some cases, and code using this can't know, so have fun. Many things are broken, but this makes e.g. vo_gpu mostly work. My general opinion about BE computers is that you should get a better computer, you can get one for free from any garbage dump.
This commit is contained in:
parent
1ffa83ea92
commit
fd9c570f22
@ -200,6 +200,7 @@ static void fill_pixdesc_layout(struct mp_imgfmt_desc *desc,
|
|||||||
// explicitly marks big endian formats => don't need to guess whether a
|
// explicitly marks big endian formats => don't need to guess whether a
|
||||||
// format is little endian, or not affected by byte order.
|
// format is little endian, or not affected by byte order.
|
||||||
bool is_be = pd->flags & AV_PIX_FMT_FLAG_BE;
|
bool is_be = pd->flags & AV_PIX_FMT_FLAG_BE;
|
||||||
|
bool is_ne = MP_SELECT_LE_BE(false, true) == is_be;
|
||||||
|
|
||||||
// Packed sub-sampled YUV is very... special.
|
// Packed sub-sampled YUV is very... special.
|
||||||
bool is_packed_ss_yuv = pd->log2_chroma_w && !pd->log2_chroma_h &&
|
bool is_packed_ss_yuv = pd->log2_chroma_w && !pd->log2_chroma_h &&
|
||||||
@ -283,19 +284,20 @@ static void fill_pixdesc_layout(struct mp_imgfmt_desc *desc,
|
|||||||
// representable, because endian_shift is for all planes).
|
// representable, because endian_shift is for all planes).
|
||||||
// As a heuristic, assume that if any components share a byte, the whole
|
// As a heuristic, assume that if any components share a byte, the whole
|
||||||
// pixel is read as a single memory access and endian swapped at once.
|
// pixel is read as a single memory access and endian swapped at once.
|
||||||
int endian_size = 8;
|
int access_size = 8;
|
||||||
if (is_be && plane_bits > 8) {
|
if (plane_bits > 8) {
|
||||||
if (any_shared_bytes) {
|
if (any_shared_bytes) {
|
||||||
endian_size = plane_bits;
|
access_size = plane_bits;
|
||||||
if (word != endian_size) {
|
if (is_be && word != access_size) {
|
||||||
// Before: offset = 8*byte_offset (with word bits of data)
|
// Before: offset = 8*byte_offset (with word bits of data)
|
||||||
// After: offset = bit_offset into swapped endian_size word
|
// After: offset = bit_offset into swapped endian_size word
|
||||||
offset = endian_size - word - offset;
|
offset = access_size - word - offset;
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
endian_size = word;
|
access_size = word;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
int endian_size = (access_size && !is_ne) ? access_size : 8;
|
||||||
int endian_shift = mp_log2(endian_size) - 3;
|
int endian_shift = mp_log2(endian_size) - 3;
|
||||||
if (!MP_IS_POWER_OF_2(endian_size) || endian_shift < 0 || endian_shift > 3)
|
if (!MP_IS_POWER_OF_2(endian_size) || endian_shift < 0 || endian_shift > 3)
|
||||||
goto fail;
|
goto fail;
|
||||||
@ -458,12 +460,13 @@ static bool mp_imgfmt_get_desc_from_pixdesc(int mpfmt, struct mp_imgfmt_desc *ou
|
|||||||
desc.align_x = 8 / desc.bpp[0]; // expect power of 2
|
desc.align_x = 8 / desc.bpp[0]; // expect power of 2
|
||||||
|
|
||||||
// Very heuristical.
|
// Very heuristical.
|
||||||
bool is_be = desc.endian_shift > 0;
|
bool is_ne = !desc.endian_shift;
|
||||||
bool need_endian = (desc.comps[0].size % 8u && desc.bpp[0] > 8) ||
|
bool need_endian = (desc.comps[0].size % 8u && desc.bpp[0] > 8) ||
|
||||||
desc.comps[0].size > 8;
|
desc.comps[0].size > 8;
|
||||||
|
|
||||||
if (need_endian) {
|
if (need_endian) {
|
||||||
desc.flags |= is_be ? MP_IMGFLAG_BE : MP_IMGFLAG_LE;
|
bool is_le = MP_SELECT_LE_BE(is_ne, !is_ne);
|
||||||
|
desc.flags |= is_le ? MP_IMGFLAG_LE : MP_IMGFLAG_BE;
|
||||||
} else {
|
} else {
|
||||||
desc.flags |= MP_IMGFLAG_LE | MP_IMGFLAG_BE;
|
desc.flags |= MP_IMGFLAG_LE | MP_IMGFLAG_BE;
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user