Merge pull request #59324 from Zylann/fix_fvec_array_binary_load
Fix loading binary resources with float=64
This commit is contained in:
commit
bf153b82c7
1 changed files with 58 additions and 41 deletions
|
@ -101,6 +101,50 @@ void ResourceLoaderBinary::_advance_padding(uint32_t p_len) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static Error read_reals(real_t *dst, FileAccess &f, size_t count) {
|
||||||
|
if (f.real_is_double) {
|
||||||
|
if (sizeof(real_t) == 8) {
|
||||||
|
// Ideal case with double-precision
|
||||||
|
f.get_buffer((uint8_t *)dst, count * sizeof(double));
|
||||||
|
#ifdef BIG_ENDIAN_ENABLED
|
||||||
|
{
|
||||||
|
uint64_t *dst = (uint64_t *)dst;
|
||||||
|
for (size_t i = 0; i < count; i++) {
|
||||||
|
dst[i] = BSWAP64(dst[i]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
} else if (sizeof(real_t) == 4) {
|
||||||
|
// May be slower, but this is for compatibility. Eventually the data should be converted.
|
||||||
|
for (size_t i = 0; i < count; ++i) {
|
||||||
|
dst[i] = f.get_double();
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
ERR_FAIL_V_MSG(ERR_UNAVAILABLE, "real_t size is neither 4 nor 8!");
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if (sizeof(real_t) == 4) {
|
||||||
|
// Ideal case with float-precision
|
||||||
|
f.get_buffer((uint8_t *)dst, count * sizeof(float));
|
||||||
|
#ifdef BIG_ENDIAN_ENABLED
|
||||||
|
{
|
||||||
|
uint32_t *dst = (uint32_t *)dst;
|
||||||
|
for (size_t i = 0; i < count; i++) {
|
||||||
|
dst[i] = BSWAP32(dst[i]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
} else if (sizeof(real_t) == 8) {
|
||||||
|
for (size_t i = 0; i < count; ++i) {
|
||||||
|
dst[i] = f.get_float();
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
ERR_FAIL_V_MSG(ERR_UNAVAILABLE, "real_t size is neither 4 nor 8!");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return OK;
|
||||||
|
}
|
||||||
|
|
||||||
StringName ResourceLoaderBinary::_get_string() {
|
StringName ResourceLoaderBinary::_get_string() {
|
||||||
uint32_t id = f->get_32();
|
uint32_t id = f->get_32();
|
||||||
if (id & 0x80000000) {
|
if (id & 0x80000000) {
|
||||||
|
@ -528,21 +572,9 @@ Error ResourceLoaderBinary::parse_variant(Variant &r_v) {
|
||||||
Vector<Vector2> array;
|
Vector<Vector2> array;
|
||||||
array.resize(len);
|
array.resize(len);
|
||||||
Vector2 *w = array.ptrw();
|
Vector2 *w = array.ptrw();
|
||||||
if (sizeof(Vector2) == 8) {
|
static_assert(sizeof(Vector2) == 2 * sizeof(real_t));
|
||||||
f->get_buffer((uint8_t *)w, len * sizeof(real_t) * 2);
|
const Error err = read_reals(reinterpret_cast<real_t *>(w), *f, len * 2);
|
||||||
#ifdef BIG_ENDIAN_ENABLED
|
ERR_FAIL_COND_V(err != OK, err);
|
||||||
{
|
|
||||||
uint32_t *ptr = (uint32_t *)w.ptr();
|
|
||||||
for (int i = 0; i < len * 2; i++) {
|
|
||||||
ptr[i] = BSWAP32(ptr[i]);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#endif
|
|
||||||
|
|
||||||
} else {
|
|
||||||
ERR_FAIL_V_MSG(ERR_UNAVAILABLE, "Vector2 size is NOT 8!");
|
|
||||||
}
|
|
||||||
|
|
||||||
r_v = array;
|
r_v = array;
|
||||||
|
|
||||||
|
@ -553,21 +585,9 @@ Error ResourceLoaderBinary::parse_variant(Variant &r_v) {
|
||||||
Vector<Vector3> array;
|
Vector<Vector3> array;
|
||||||
array.resize(len);
|
array.resize(len);
|
||||||
Vector3 *w = array.ptrw();
|
Vector3 *w = array.ptrw();
|
||||||
if (sizeof(Vector3) == 12) {
|
static_assert(sizeof(Vector3) == 3 * sizeof(real_t));
|
||||||
f->get_buffer((uint8_t *)w, len * sizeof(real_t) * 3);
|
const Error err = read_reals(reinterpret_cast<real_t *>(w), *f, len * 3);
|
||||||
#ifdef BIG_ENDIAN_ENABLED
|
ERR_FAIL_COND_V(err != OK, err);
|
||||||
{
|
|
||||||
uint32_t *ptr = (uint32_t *)w.ptr();
|
|
||||||
for (int i = 0; i < len * 3; i++) {
|
|
||||||
ptr[i] = BSWAP32(ptr[i]);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#endif
|
|
||||||
|
|
||||||
} else {
|
|
||||||
ERR_FAIL_V_MSG(ERR_UNAVAILABLE, "Vector3 size is NOT 12!");
|
|
||||||
}
|
|
||||||
|
|
||||||
r_v = array;
|
r_v = array;
|
||||||
|
|
||||||
|
@ -578,22 +598,19 @@ Error ResourceLoaderBinary::parse_variant(Variant &r_v) {
|
||||||
Vector<Color> array;
|
Vector<Color> array;
|
||||||
array.resize(len);
|
array.resize(len);
|
||||||
Color *w = array.ptrw();
|
Color *w = array.ptrw();
|
||||||
if (sizeof(Color) == 16) {
|
// Colors always use `float` even with double-precision support enabled
|
||||||
f->get_buffer((uint8_t *)w, len * sizeof(real_t) * 4);
|
static_assert(sizeof(Color) == 4 * sizeof(float));
|
||||||
|
f->get_buffer((uint8_t *)w, len * sizeof(float) * 4);
|
||||||
#ifdef BIG_ENDIAN_ENABLED
|
#ifdef BIG_ENDIAN_ENABLED
|
||||||
{
|
{
|
||||||
uint32_t *ptr = (uint32_t *)w.ptr();
|
uint32_t *ptr = (uint32_t *)w.ptr();
|
||||||
for (int i = 0; i < len * 4; i++) {
|
for (int i = 0; i < len * 4; i++) {
|
||||||
ptr[i] = BSWAP32(ptr[i]);
|
ptr[i] = BSWAP32(ptr[i]);
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
} else {
|
|
||||||
ERR_FAIL_V_MSG(ERR_UNAVAILABLE, "Color size is NOT 16!");
|
|
||||||
}
|
|
||||||
|
|
||||||
r_v = array;
|
r_v = array;
|
||||||
} break;
|
} break;
|
||||||
default: {
|
default: {
|
||||||
|
|
Loading…
Add table
Reference in a new issue