From 1c148447caef80f810f39874e01687d43514d0f5 Mon Sep 17 00:00:00 2001 From: David Tschumperle Date: Thu, 19 Sep 2024 09:55:00 +0200 Subject: [PATCH 1/4] Auto-commit for release 3.4.3_pre --- html/header.html | 2 +- html/header_doxygen.html | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/html/header.html b/html/header.html index 25b4127a..c0570507 100644 --- a/html/header.html +++ b/html/header.html @@ -23,7 +23,7 @@
Logo

- Latest stable version: 3.4.2        Current pre-release: 3.4.3 (2024/09/17) + Latest stable version: 3.4.2        Current pre-release: 3.4.3 (2024/09/19)


diff --git a/html/header_doxygen.html b/html/header_doxygen.html index 2b91654d..3e0ae801 100644 --- a/html/header_doxygen.html +++ b/html/header_doxygen.html @@ -26,7 +26,7 @@
Logo

- Latest stable version: 3.4.2        Current pre-release: 3.4.3 (2024/09/17) + Latest stable version: 3.4.2        Current pre-release: 3.4.3 (2024/09/19)


From 417609f6221e0f62eb3ba642015b4168425b3f16 Mon Sep 17 00:00:00 2001 From: apocelipes Date: Thu, 19 Sep 2024 19:04:05 +0800 Subject: [PATCH 2/4] Only load RGB data if a webp image does not have an alpha channel --- CImg.h | 48 +++++++++++++++++++++++++++++++++--------------- 1 file changed, 33 insertions(+), 15 deletions(-) diff --git a/CImg.h b/CImg.h index a204205d..210d6dad 100644 --- a/CImg.h +++ b/CImg.h @@ -58231,36 +58231,54 @@ namespace cimg_library { } CImg buffer(data_size); cimg::fread(buffer._data, buffer._width, file); - int width = 0, height = 0; - if (!WebPGetInfo(buffer._data, data_size, &width, &height)) { - cimg::fclose(file); + cimg::fclose(file); + + WebPDecoderConfig config; + if (!WebPInitDecoderConfig(&config)) + throw CImgIOException(_cimg_instance + "load_webp(): Failed to init WebP decoder config.", + cimg_instance); + + if (WebPGetFeatures(buffer._data, data_size, &config.input) != VP8_STATUS_OK) + throw CImgIOException(_cimg_instance + "load_webp(): Failed to get image meta info of '%s'.", + cimg_instance, + filename); + + if (config.input.has_animation) throw CImgIOException(_cimg_instance - "load_webp(): Failed to get image width/height '%s'.", + "load_webp(): Does not support animated WebP '%s'.", cimg_instance, filename); + + int width = config.input.width, height = config.input.height; + if (config.input.has_alpha) { + config.output.colorspace = MODE_RGBA; + assign(width,height,1,4); + } else { + config.output.colorspace = MODE_RGB; + assign(width,height,1,3); } - assign(width,height,1,4); - unsigned char *imgData = WebPDecodeRGBA(buffer._data, data_size, NULL, NULL); - if (!imgData) { - cimg::fclose(file); + if (WebPDecode(buffer._data, data_size, &config) != VP8_STATUS_OK) throw CImgIOException(_cimg_instance "load_webp(): Failed to decode image '%s'.", cimg_instance, filename); - } + + uint8_t *imgData = config.output.u.RGBA.rgba; T *ptr_r = _data, *ptr_g = _data + 1UL*width*height, - *ptr_b = _data + 2UL*width*height, *ptr_a = _data + 3UL*width*height; + *ptr_b = _data + 2UL*width*height; + T *ptr_a = _spectrum==3 ? NULL : _data + 3UL*width*height; cimg_forY(*this,y) { - const unsigned char *ptrs = (unsigned char*)&imgData[y*width*4]; + const unsigned char *ptrs = (unsigned char*)&imgData[y*width*_spectrum]; cimg_forX(*this,x) { *(ptr_r++) = (T)*(ptrs++); *(ptr_g++) = (T)*(ptrs++); *(ptr_b++) = (T)*(ptrs++); - *(ptr_a++) = (T)*(ptrs++); + if (ptr_a) *(ptr_a++) = (T)*(ptrs++); } } - WebPFree(imgData); - cimg::fclose(file); + WebPFreeDecBuffer(&config.output); return *this; #endif } @@ -60304,6 +60322,7 @@ namespace cimg_library { cimg::unused(quality); return save_other(filename); #else + std::FILE *file = cimg::fopen(filename, "wb"); CImg rgbaBuffer(size()); T *ptr_r = _data, *ptr_g = _data + 1UL*_width*_height, *ptr_b = _data + 2UL*_width*_height, *ptr_a = _spectrum==3?NULL:_data + 3UL*_width*_height; @@ -60330,7 +60349,6 @@ namespace cimg_library { cimg_instance, filename); } - std::FILE *file = cimg::fopen(filename, "wb"); cimg::fwrite(imgData, size, file); cimg::fclose(file); WebPFree(imgData); From 7904da33c164db1cf9dc63f5ddf9fdf48adc3ecd Mon Sep 17 00:00:00 2001 From: David Tschumperle Date: Sat, 21 Sep 2024 08:52:50 +0200 Subject: [PATCH 3/4] cimg_math_parser: Make function 'find()' works with 'nan' values when image index is specified. --- CImg.h | 26 ++++++++++++++++++++------ 1 file changed, 20 insertions(+), 6 deletions(-) diff --git a/CImg.h b/CImg.h index 210d6dad..074ef96a 100644 --- a/CImg.h +++ b/CImg.h @@ -26708,12 +26708,18 @@ namespace cimg_library { // Forward search if (step>0) { - while (ptr::is_nan(val)) + while (ptr::is_nan(*ptr)) ptr+=step; + else + while (ptr=ptre?-1.:(double)(ptr - ptrb); } // Backward search. - while (ptr>=ptrb && (double)*ptr!=val) ptr+=step; + if (cimg::type::is_nan(val)) + while (ptr>=ptrb && !cimg::type::is_nan(*ptr)) ptr+=step; + else + while (ptr>=ptrb && (double)*ptr!=val) ptr+=step; return ptr0) { do { - while (ptr1::is_nan(*ptr2b)) + while (ptr1::is_nan(*ptr1)) ptr1+=step; + else + while (ptr1=ptr1e) return -1.; p1 = ptr1 + 1; p2 = ptr2b + 1; - while (p1::is_nan(*p1) && cimg::type::is_nan(*p2)) || *p1==*p2)) { ++p1; ++p2; } } while (p2=ptr1b && *ptr1!=*ptr2b) ptr1+=step; + if (cimg::type::is_nan(*ptr2b)) + while (ptr1>=ptr1b && !cimg::type::is_nan(*ptr1)) ptr1+=step; + else + while (ptr1>=ptr1b && *ptr1!=*ptr2b) ptr1+=step; if (ptr1::is_nan(*p1) && cimg::type::is_nan(*p2)) || *p1==*p2)) { ++p1; ++p2; } } while (p2=ptr1b); return p2 Date: Wed, 25 Sep 2024 08:52:23 +0200 Subject: [PATCH 4/4] CImg::mp_vector_eq(): Recode vector comparison to better handle NaN values. --- CImg.h | 97 +++++++++++++++++++++++++++------------------------------- 1 file changed, 45 insertions(+), 52 deletions(-) diff --git a/CImg.h b/CImg.h index 074ef96a..b831c173 100644 --- a/CImg.h +++ b/CImg.h @@ -29048,61 +29048,54 @@ namespace cimg_library { const double *ptr1 = &_mp_arg(2) + 1, *ptr2 = &_mp_arg(4) + 1; - unsigned int p1 = (unsigned int)mp.opcode[3], p2 = (unsigned int)mp.opcode[5], n; - const int N = (int)_mp_arg(6); + unsigned int + siz1 = (unsigned int)mp.opcode[3], + siz2 = (unsigned int)mp.opcode[5], + n; + int N = (int)_mp_arg(6); const bool case_sensitive = (bool)_mp_arg(7); + if (!N) return 1; + if (N<0) { // Compare all values + if (siz1>0 && siz2>0 && siz1!=siz2) return 0; + N = std::max(siz1,siz2); + } + + // Compare first N values. bool still_equal = true; - double value; - if (!N) return true; - - // Compare all values. - if (N<0) { - if (p1>0 && p2>0) { // Vector == vector - if (p1!=p2) return false; - if (case_sensitive) - while (still_equal && p1--) still_equal = *(ptr1++)==*(ptr2++); - else - while (still_equal && p1--) - still_equal = cimg::lowercase(*(ptr1++))==cimg::lowercase(*(ptr2++)); - return still_equal; - } else if (p1>0 && !p2) { // Vector == scalar - value = _mp_arg(4); - if (!case_sensitive) value = cimg::lowercase(value); - while (still_equal && p1--) still_equal = *(ptr1++)==value; - return still_equal; - } else if (!p1 && p2>0) { // Scalar == vector - value = _mp_arg(2); - if (!case_sensitive) value = cimg::lowercase(value); - while (still_equal && p2--) still_equal = *(ptr2++)==value; - return still_equal; - } else { // Scalar == scalar - if (case_sensitive) return _mp_arg(2)==_mp_arg(4); - else return cimg::lowercase(_mp_arg(2))==cimg::lowercase(_mp_arg(4)); - } - } - - // Compare only first N values. - if (p1>0 && p2>0) { // Vector == vector - n = cimg::min((unsigned int)N,p1,p2); + if (siz1>0 && siz2>0) { // Vector == vector + n = cimg::min((unsigned int)N,siz1,siz2); if (case_sensitive) - while (still_equal && n--) still_equal = *(ptr1++)==(*ptr2++); + while (still_equal && n--) { + still_equal = (cimg::type::is_nan(*ptr1) && cimg::type::is_nan(*ptr2)) || + *ptr1==*ptr2; + ++ptr1; ++ptr2; + } else - while (still_equal && n--) still_equal = cimg::lowercase(*(ptr1++))==cimg::lowercase(*(ptr2++)); + while (still_equal && n--) { + still_equal = (cimg::type::is_nan(*ptr1) && cimg::type::is_nan(*ptr2)) || + cimg::lowercase(*ptr1)==cimg::lowercase(*ptr2); + ++ptr1; ++ptr2; + } return still_equal; - } else if (p1>0 && !p2) { // Vector == scalar - n = std::min((unsigned int)N,p1); - value = _mp_arg(4); - if (!case_sensitive) value = cimg::lowercase(value); - while (still_equal && n--) still_equal = *(ptr1++)==value; + } else if (siz1>0 && !siz2) { // Vector == scalar + n = std::min((unsigned int)N,siz1); + const double value = case_sensitive?_mp_arg(4):cimg::lowercase(_mp_arg(4)); + if (cimg::type::is_nan(value)) + while (still_equal && n--) still_equal = cimg::type::is_nan(*(ptr1++)); + else + while (still_equal && n--) still_equal = *(ptr1++)==value; return still_equal; - } else if (!p1 && p2>0) { // Scalar == vector - n = std::min((unsigned int)N,p2); - value = _mp_arg(2); - if (!case_sensitive) value = cimg::lowercase(value); - while (still_equal && n--) still_equal = *(ptr2++)==value; + } else if (!siz1 && siz2>0) { // Scalar == vector + n = std::min((unsigned int)N,siz2); + const double value = case_sensitive?_mp_arg(2):cimg::lowercase(_mp_arg(2)); + if (cimg::type::is_nan(value)) + while (still_equal && siz2--) still_equal = cimg::type::is_nan(*(ptr2++)); + else + while (still_equal && n--) still_equal = *(ptr2++)==value; return still_equal; } // Scalar == scalar if (case_sensitive) return _mp_arg(2)==_mp_arg(4); + else if (cimg::type::is_nan(_mp_arg(2))) return cimg::type::is_nan(_mp_arg(4)); return cimg::lowercase(_mp_arg(2))==cimg::lowercase(_mp_arg(4)); } @@ -29267,18 +29260,18 @@ namespace cimg_library { static double mp_vector_resize(_cimg_math_parser& mp) { double *const ptrd = &_mp_arg(1) + 1; - const unsigned int siz = (unsigned int)mp.opcode[2], p2 = (unsigned int)mp.opcode[4]; + const unsigned int target_siz = (unsigned int)mp.opcode[2], initial_siz = (unsigned int)mp.opcode[4]; const int interpolation = (int)_mp_arg(5), boundary_conditions = (int)_mp_arg(6); - if (p2) { // Resize vector + if (initial_siz) { // Resize vector const double *const ptrs = &_mp_arg(3) + 1; - CImg(ptrd,siz,1,1,1,true) = CImg(ptrs,p2,1,1,1,true). - get_resize(siz,1,1,1,interpolation,boundary_conditions); + CImg(ptrd,target_siz,1,1,1,true) = CImg(ptrs,initial_siz,1,1,1,true). + get_resize(target_siz,1,1,1,interpolation,boundary_conditions); } else { // Resize scalar const double value = _mp_arg(3); - CImg(ptrd,siz,1,1,1,true) = CImg(1,1,1,1,value).resize(siz,1,1,1,interpolation, - boundary_conditions); + CImg(ptrd,target_siz,1,1,1,true) = CImg(1,1,1,1,value). + resize(target_siz,1,1,1,interpolation,boundary_conditions); } return cimg::type::nan(); }