+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::add_output(const ImageFormat &format, OutputAlphaFormat alpha_format)
+{
+ chain.add_output(format, alpha_format);
+ output_added = true;
+}
+
+void EffectChainTester::add_ycbcr_output(const ImageFormat &format, OutputAlphaFormat alpha_format, const YCbCrFormat &ycbcr_format)
+{
+ chain.add_ycbcr_output(format, alpha_format, ycbcr_format);
+ output_added = true;
+}
+
+void EffectChainTester::finalize_chain(Colorspace color_space, GammaCurve gamma_curve, OutputAlphaFormat alpha_format)