X-Git-Url: http://git.tdb.fi/?p=libs%2Fgl.git;a=blobdiff_plain;f=source%2Fbuilders%2Fsequencebuilder.cpp;h=65aeaa13d01a02d1dfa12198dfb4ec8fc4f39be6;hp=317b68846126b3330a5e5592dfbc3deee01cf2af;hb=d16d28d2ccf7c6255204f02975834f713ff1df08;hpb=3ac3a51c623271da815c8ee60c484445871753bf diff --git a/source/builders/sequencebuilder.cpp b/source/builders/sequencebuilder.cpp index 317b6884..65aeaa13 100644 --- a/source/builders/sequencebuilder.cpp +++ b/source/builders/sequencebuilder.cpp @@ -1,10 +1,12 @@ #include #include +#include +#include "device.h" #include "error.h" -#include "renderbuffer.h" #include "sequence.h" #include "sequencebuilder.h" #include "sequencetemplate.h" +#include "view.h" using namespace std; @@ -14,13 +16,11 @@ namespace GL { SequenceBuilder::SequenceBuilder(const SequenceTemplate &t): tmpl(t) { - const vector &steps = tmpl.get_steps(); - for(vector::const_iterator i=steps.begin(); i!=steps.end(); ++i) - renderables[i->slot_name] = i->default_renderable; - const vector &postprocs = tmpl.get_postprocessors(); - for(SequenceTemplate::PostProcessorArray::const_iterator i=postprocs.begin(); i!=postprocs.end(); ++i) - if(!i->slot_name.empty()) - postprocessors[i->slot_name] = 0; + for(const SequenceTemplate::Renderable &r: tmpl.get_renderables()) + renderables[r.slot_name] = r.renderable; + for(const SequenceTemplate::PostProcessor &p: tmpl.get_postprocessors()) + if(!p.slot_name.empty()) + postprocessors[p.slot_name] = 0; } void SequenceBuilder::set_renderable(const string &name, Renderable &rend) @@ -33,66 +33,150 @@ void SequenceBuilder::set_postprocessor(const string &name, PostProcessor &pproc get_item(postprocessors, name) = &pproc; } +void SequenceBuilder::set_debug_name(const string &name) +{ +#ifdef DEBUG + debug_name = name; +#else + (void)name; +#endif +} + void SequenceBuilder::build(Sequence &sequence) const { - sequence.set_hdr(tmpl.get_hdr()); - sequence.set_alpha(tmpl.get_alpha()); - unsigned samples = min(tmpl.get_maximum_multisample(), Renderbuffer::get_max_samples()); - if(samples final_renderables = renderables; + vector sequences; + for(const SequenceTemplate::Renderable &r: tmpl.get_renderables()) + { + Renderable *&ptr = final_renderables[r.slot_name]; + if(!ptr) + { + if(r.effect_template) + { + Effect *effect = r.effect_template->create(final_renderables); +#ifdef DEBUG + if(!debug_name.empty()) + effect->set_debug_name(format("%s/%s", debug_name, r.slot_name)); +#endif + ptr = effect; + } + else if(r.sequence_template) + { + Sequence *seq = new Sequence; +#ifdef DEBUG + if(!debug_name.empty()) + seq->set_debug_name(format("%s/%s", debug_name, r.slot_name)); +#endif + ptr = seq; + sequences.push_back(&r); + } + else + throw invalid_operation("SequenceBuilder::build"); - sequence.set_multisample(samples); + sequence.add_owned(ptr); + } + } + + for(const SequenceTemplate::Renderable *s: sequences) + { + SequenceBuilder bld(*s->sequence_template); + for(const auto &kvp: s->sequence_renderables) + if(Renderable *r = get_item(final_renderables, kvp.second)) + bld.set_renderable(kvp.first, *r); + bld.build(*static_cast(get_item(final_renderables, s->slot_name))); + } + + sequence.set_clear_enabled(tmpl.is_clear_enabled()); + if(tmpl.is_clear_enabled()) + { + sequence.set_clear_colors(tmpl.get_clear_colors()); + sequence.set_clear_depth(tmpl.get_clear_depth()); + sequence.set_clear_stencil(tmpl.get_clear_stencil()); + } - const vector &steps = tmpl.get_steps(); - for(vector::const_iterator i=steps.begin(); i!=steps.end(); ++i) + for(const SequenceTemplate::Step &s: tmpl.get_steps()) { - Renderable *renderable = get_item(renderables, i->slot_name); - if(!renderable) - continue; - - Sequence::Step &step = sequence.add_step(i->tag, *renderable); - step.set_blend(i->blend.get()); - step.set_depth_test(i->depth_test.get()); - step.set_lighting(i->lighting); + // Existence is already ensured above + Renderable &renderable = *get_item(final_renderables, s.renderable_name); + Sequence::Step &step = sequence.add_step(s.tag, renderable); + step.set_depth_test(s.depth_test); + step.set_stencil_test(s.stencil_test); + step.set_lighting(s.lighting); } - const SequenceTemplate::PostProcessorArray &postprocs = tmpl.get_postprocessors(); - for(SequenceTemplate::PostProcessorArray::const_iterator i=postprocs.begin(); i!=postprocs.end(); ++i) +#ifdef DEBUG + unsigned index = 0; +#endif + for(const SequenceTemplate::PostProcessor &p: tmpl.get_postprocessors()) { PostProcessor *proc = 0; - if(!i->slot_name.empty()) - proc = get_item(postprocessors, i->slot_name); - if(proc) - sequence.add_postprocessor(*proc); - else if(i->postprocessor_template) + if(!p.slot_name.empty()) + proc = get_item(postprocessors, p.slot_name); + if(!proc) { - proc = i->postprocessor_template->create(sequence.get_width(), sequence.get_height()); - if(proc) - sequence.add_postprocessor_owned(proc); + if(p.postprocessor_template) + { + proc = p.postprocessor_template->create(sequence.get_width(), sequence.get_height()); +#ifdef DEBUG + if(!debug_name.empty()) + proc->set_debug_name(format("%s/%d.pproc", debug_name, index++)); +#endif + } + else + throw invalid_operation("SequenceBuilder::build"); + + sequence.add_owned(proc); } + + sequence.add_postprocessor(*proc); } } +Sequence *SequenceBuilder::build() const +{ + RefPtr sequence = new Sequence(); + build(*sequence); + return sequence.release(); +} + Sequence *SequenceBuilder::build(unsigned w, unsigned h) const { - RefPtr sequence = new Sequence(w, h); + RefPtr sequence = new Sequence(w, h, create_frame_format()); build(*sequence); return sequence.release(); } Sequence *SequenceBuilder::build(const View &view) const { - RefPtr sequence = new Sequence(view); + RefPtr sequence = new Sequence(view.get_width(), view.get_height(), create_frame_format()); build(*sequence); return sequence.release(); } Sequence *SequenceBuilder::build(const Framebuffer &fbo) const { - RefPtr sequence = new Sequence(fbo); + RefPtr sequence = new Sequence(fbo.get_width(), fbo.get_height(), create_frame_format()); build(*sequence); return sequence.release(); } +FrameFormat SequenceBuilder::create_frame_format() const +{ + unsigned samples = min(tmpl.get_maximum_multisample(), Device::get_current().get_info().limits.max_samples); + if(samples