+void EffectChainTester::run(unsigned char *out_data, GLenum format, Colorspace color_space, GammaCurve gamma_curve, OutputAlphaFormat alpha_format)
+{
+ if (!finalized) {
+ finalize_chain(color_space, gamma_curve, alpha_format);
+ }
+
+ chain.render_to_fbo(fbo, width, height);
+
+ glBindFramebuffer(GL_FRAMEBUFFER, fbo);
+ check_error();
+ if (!epoxy_is_desktop_gl() && (format == GL_RED || format == GL_BLUE || format == GL_ALPHA)) {
+ // GLES will only read GL_RGBA.
+ unsigned char *temp = new unsigned char[width * height * 4];
+ glReadPixels(0, 0, width, height, GL_RGBA, GL_UNSIGNED_BYTE, temp);
+ check_error();
+ if (format == GL_ALPHA) {
+ for (unsigned i = 0; i < width * height; ++i) {
+ out_data[i] = temp[i * 4 + 3];
+ }
+ } else if (format == GL_BLUE) {
+ for (unsigned i = 0; i < width * height; ++i) {
+ out_data[i] = temp[i * 4 + 2];
+ }
+ } else {
+ for (unsigned i = 0; i < width * height; ++i) {
+ out_data[i] = temp[i * 4];
+ }
+ }
+ delete[] temp;
+ } else {
+ glReadPixels(0, 0, width, height, format, GL_UNSIGNED_BYTE, out_data);
+ check_error();
+ }
+
+ if (format == GL_RGBA) {
+ width *= 4;
+ }
+
+ vertical_flip(out_data, width, height);
+}
+
+void EffectChainTester::finalize_chain(Colorspace color_space, GammaCurve gamma_curve, OutputAlphaFormat alpha_format)
+{
+ assert(!finalized);
+ ImageFormat image_format;
+ image_format.color_space = color_space;
+ image_format.gamma_curve = gamma_curve;
+ chain.add_output(image_format, alpha_format);
+ chain.finalize();
+ finalized = true;
+}
+
+void expect_equal(const float *ref, const float *result, unsigned width, unsigned height, float largest_difference_limit, float rms_limit)