OSSIA
Open Scenario System for Interactive Application
Loading...
Searching...
No Matches
faust_utils.hpp
1#pragma once
2#include <ossia/dataflow/graph_node.hpp>
3#include <ossia/dataflow/port.hpp>
5
6#include <faust/gui/UI.h>
7#include <libremidi/message.hpp>
8#include <libremidi/ump_events.hpp>
9
10#include <faust/dsp/poly-llvm-dsp.h>
11
12namespace ossia::nodes
13{
14
15template <typename T>
16struct faust_setup_ui : UI
17{
18 faust_setup_ui(T& self) { }
19};
20
21template <typename Node, bool Synth>
22struct faust_exec_ui final : UI
23{
24 Node& fx;
25 faust_exec_ui(Node& n)
26 : fx{n}
27 {
28 }
29
30 void addButton(const char* label, FAUSTFLOAT* zone) override
31 {
32 if constexpr(Synth)
33 {
34 using namespace std::literals;
35 if(label == "Panic"sv || label == "gate"sv)
36 return;
37 }
38
39 fx.root_inputs().push_back(new ossia::value_inlet);
40 fx.controls.push_back(
41 {fx.root_inputs().back()->template target<ossia::value_port>(), zone});
42 }
43
44 void addCheckButton(const char* label, FAUSTFLOAT* zone) override
45 {
46 addButton(label, zone);
47 }
48
49 void addVerticalSlider(
50 const char* label, FAUSTFLOAT* zone, FAUSTFLOAT init, FAUSTFLOAT min,
51 FAUSTFLOAT max, FAUSTFLOAT step) override
52 {
53 if constexpr(Synth)
54 {
55 using namespace std::literals;
56 if(label == "gain"sv || label == "freq"sv || label == "sustain"sv)
57 return;
58 }
59 fx.root_inputs().push_back(new ossia::value_inlet);
60 fx.controls.push_back(
61 {fx.root_inputs().back()->template target<ossia::value_port>(), zone});
62 }
63
64 void addHorizontalSlider(
65 const char* label, FAUSTFLOAT* zone, FAUSTFLOAT init, FAUSTFLOAT min,
66 FAUSTFLOAT max, FAUSTFLOAT step) override
67 {
68 addVerticalSlider(label, zone, init, min, max, step);
69 }
70
71 void addNumEntry(
72 const char* label, FAUSTFLOAT* zone, FAUSTFLOAT init, FAUSTFLOAT min,
73 FAUSTFLOAT max, FAUSTFLOAT step) override
74 {
75 addVerticalSlider(label, zone, init, min, max, step);
76 }
77
78 void addHorizontalBargraph(
79 const char* label, FAUSTFLOAT* zone, FAUSTFLOAT min, FAUSTFLOAT max) override
80 {
81 fx.root_outputs().push_back(new ossia::value_outlet);
82 fx.displays.push_back(
83 {fx.root_outputs().back()->template target<ossia::value_port>(), zone});
84 }
85
86 void addVerticalBargraph(
87 const char* label, FAUSTFLOAT* zone, FAUSTFLOAT min, FAUSTFLOAT max) override
88 {
89 addHorizontalBargraph(label, zone, min, max);
90 }
91
92 void openTabBox(const char* label) override { }
93 void openHorizontalBox(const char* label) override { }
94 void openVerticalBox(const char* label) override { }
95 void closeBox() override { }
96 void declare(FAUSTFLOAT* zone, const char* key, const char* val) override { }
97 void
98 addSoundfile(const char* label, const char* filename, Soundfile** sf_zone) override
99 {
100 }
101};
102
103template <typename Clone>
104struct faust_exec_ui_clone final : ::UI
105{
106 Clone& self;
107 int i = 0;
108 int o = 0;
109 faust_exec_ui_clone(Clone& s)
110 : self{s}
111 {
112 }
113
114 void addButton(const char* label, FAUSTFLOAT* zone) override
115 {
116 *zone = *self.controls[i].second;
117 self.controls[i++].second = zone;
118 }
119
120 void addCheckButton(const char* label, FAUSTFLOAT* zone) override
121 {
122 addButton(label, zone);
123 }
124
125 void addVerticalSlider(
126 const char* label, FAUSTFLOAT* zone, FAUSTFLOAT init, FAUSTFLOAT min,
127 FAUSTFLOAT max, FAUSTFLOAT step) override
128 {
129 *zone = *self.controls[i].second;
130 self.controls[i++].second = zone;
131 }
132
133 void addHorizontalSlider(
134 const char* label, FAUSTFLOAT* zone, FAUSTFLOAT init, FAUSTFLOAT min,
135 FAUSTFLOAT max, FAUSTFLOAT step) override
136 {
137 addVerticalSlider(label, zone, init, min, max, step);
138 }
139
140 void addNumEntry(
141 const char* label, FAUSTFLOAT* zone, FAUSTFLOAT init, FAUSTFLOAT min,
142 FAUSTFLOAT max, FAUSTFLOAT step) override
143 {
144 addVerticalSlider(label, zone, init, min, max, step);
145 }
146
147 void addHorizontalBargraph(
148 const char* label, FAUSTFLOAT* zone, FAUSTFLOAT min, FAUSTFLOAT max) override
149 {
150 self.displays[o++].second = zone;
151 }
152
153 void addVerticalBargraph(
154 const char* label, FAUSTFLOAT* zone, FAUSTFLOAT min, FAUSTFLOAT max) override
155 {
156 addHorizontalBargraph(label, zone, min, max);
157 }
158
159 void openTabBox(const char* label) override { }
160 void openHorizontalBox(const char* label) override { }
161 void openVerticalBox(const char* label) override { }
162 void closeBox() override { }
163 void declare(FAUSTFLOAT* zone, const char* key, const char* val) override { }
164 void
165 addSoundfile(const char* label, const char* filename, Soundfile** sf_zone) override
166 {
167 }
168};
169
170struct faust_node_utils
171{
172 template <typename Node>
173 static void copy_controls(Node& self)
174 {
175 for(auto ctrl : self.controls)
176 {
177 auto& dat = ctrl.first->get_data();
178 if(!dat.empty())
179 {
180 *ctrl.second = ossia::convert<float>(dat.back().value);
181 }
182 }
183 }
184
185 template <typename Node>
186 static void copy_displays(Node& self, int64_t ts)
187 {
188 for(auto ctrl : self.displays)
189 {
190 ctrl.first->write_value(*ctrl.second, ts);
191 }
192 }
193
194 template <typename Node>
195 static void copy_input(
196 Node& self, int64_t d, int64_t n_in, float* inputs_, float** input_n,
197 const ossia::audio_port& audio_in)
198 {
199 // TODO offset !!!
200 for(int64_t i = 0; i < n_in; i++)
201 {
202 input_n[i] = inputs_ + i * d;
203 if(int64_t(audio_in.channels()) > i)
204 {
205 auto num_samples = std::min((int64_t)d, (int64_t)audio_in.channel(i).size());
206 for(int64_t j = 0; j < num_samples; j++)
207 {
208 input_n[i][j] = (float)audio_in.channel(i)[j];
209 }
210
211 if(d > int64_t(audio_in.channel(i).size()))
212 {
213 for(int64_t j = audio_in.channel(i).size(); j < d; j++)
214 {
215 input_n[i][j] = 0.f;
216 }
217 }
218 }
219 else
220 {
221 for(int64_t j = 0; j < d; j++)
222 {
223 input_n[i][j] = 0.f;
224 }
225 }
226 }
227 }
228 template <typename Node>
229 static void copy_input_mono(
230 Node& self, int64_t d, int64_t i, float* input,
231 const ossia::audio_channel& audio_in)
232 {
233 // TODO offset !!!
234 auto num_samples = std::min((int64_t)d, (int64_t)audio_in.size());
235 for(int64_t j = 0; j < num_samples; j++)
236 {
237 input[j] = (float)audio_in[j];
238 }
239
240 if(d > int64_t(audio_in.size()))
241 {
242 for(int64_t j = audio_in.size(); j < d; j++)
243 {
244 input[j] = 0.f;
245 }
246 }
247 }
248
249 template <typename Node>
250 static void
251 init_output(Node& self, int64_t d, int64_t n_out, float* outputs_, float** output_n)
252 {
253 for(int64_t i = 0; i < n_out; i++)
254 {
255 output_n[i] = outputs_ + i * d;
256 for(int64_t j = 0; j < d; j++)
257 {
258 output_n[i][j] = 0.f;
259 }
260 }
261 }
262
263 template <typename Node>
264 static void copy_output(
265 Node& self, int64_t d, int64_t n_out, float* outputs_, float** output_n,
266 ossia::audio_port& audio_out)
267 {
268 audio_out.set_channels(n_out);
269 for(int64_t i = 0; i < n_out; i++)
270 {
271 audio_out.channel(i).resize(d);
272 for(int64_t j = 0; j < d; j++)
273 {
274 audio_out.channel(i)[j] = (double)output_n[i][j];
275 }
276 }
277
278 // TODO handle multichannel cleanly
279 if(n_out == 1)
280 {
281 audio_out.set_channels(2);
282 audio_out.channel(1) = audio_out.channel(0);
283 }
284 }
285
286 template <typename Node, typename Dsp>
287 static void copy_midi(Node& self, Dsp& dsp, const ossia::midi_port& midi_in)
288 {
289 // TODO offset !!!
290
291 for(const libremidi::ump& mess : midi_in.messages)
292 {
293 if(mess.get_type() != libremidi::midi2::message_type::MIDI_2_CHANNEL)
294 continue;
295
296 switch(libremidi::message_type(mess.get_status_code()))
297 {
298 case libremidi::message_type::NOTE_ON: {
299 auto [channel, note, value] = libremidi::as_midi1::note_on(mess);
300 self.in_flight[note]++;
301 dsp.keyOn(channel, note, value);
302 break;
303 }
304 case libremidi::message_type::NOTE_OFF: {
305 auto [channel, note, value] = libremidi::as_midi1::note_off(mess);
306 self.in_flight[note]--;
307 dsp.keyOff(channel, note, value);
308 break;
309 }
310 case libremidi::message_type::CONTROL_CHANGE: {
311 auto [channel, index, value] = libremidi::as_midi1::control_change(mess);
312 dsp.ctrlChange(channel, index, value);
313 break;
314 }
315 case libremidi::message_type::PITCH_BEND: {
316 auto [channel, value] = libremidi::as_midi1::pitch_bend(mess);
317 dsp.pitchWheel(channel, value);
318 break;
319 }
320 default:
321 break;
322 // TODO continue...
323 }
324 }
325 }
326
327 template <typename Node, typename DspPoly>
328 void all_notes_off(Node& self, DspPoly& dsp)
329 {
330 for(int k = 0; k < 128; k++)
331 while(self.in_flight[k]-- > 0)
332 dsp.keyOff(1, k, 0);
333 }
334
336
337 template <typename Node, typename Dsp>
338 static void do_exec(
339 Node& self, Dsp& dsp, const ossia::token_request& tk,
340 const ossia::exec_state_facade& e)
341 {
342 const auto [st, d] = e.timings(tk);
343 ossia::audio_port& audio_in
344 = self.root_inputs()[0]->template cast<ossia::audio_port>();
345 ossia::audio_port& audio_out
346 = self.root_outputs()[0]->template cast<ossia::audio_port>();
347
348 const int64_t n_in = dsp.getNumInputs();
349 const int64_t n_out = dsp.getNumOutputs();
350 audio_in.set_channels(n_in);
351 audio_out.set_channels(n_out);
352
353 if constexpr(std::is_same_v<FAUSTFLOAT, float>)
354 {
355 float* inputs_ = (float*)alloca(n_in * d * sizeof(float));
356 float* outputs_ = (float*)alloca(n_out * d * sizeof(float));
357
358 float** input_n = (float**)alloca(sizeof(float*) * n_in);
359 float** output_n = (float**)alloca(sizeof(float*) * n_out);
360
361 copy_input(self, d, n_in, inputs_, input_n, audio_in);
362 init_output(self, d, n_out, outputs_, output_n);
363 dsp.compute(d, input_n, output_n);
364 copy_output(self, d, n_out, outputs_, output_n, audio_out);
365 }
366 else
367 {
368 double** input_n = (double**)alloca(sizeof(double*) * n_in);
369 double** output_n = (double**)alloca(sizeof(double*) * n_out);
370 for(int i = 0; i < n_in; i++)
371 {
372 audio_in.channel(i).resize(e.bufferSize());
373 input_n[i] = audio_in.channel(i).data() + st;
374 }
375 if(BOOST_LIKELY(st == 0 && d == e.bufferSize()))
376 {
377 for(int i = 0; i < n_out; i++)
378 {
379 audio_out.channel(i).resize(e.bufferSize(), boost::container::default_init);
380 output_n[i] = audio_out.channel(i).data() + st;
381 }
382 }
383 else
384 {
385 for(int i = 0; i < n_out; i++)
386 {
387 audio_out.channel(i).resize(e.bufferSize());
388 output_n[i] = audio_out.channel(i).data() + st;
389 }
390 }
391
392 dsp.compute(d, input_n, output_n);
393 }
394 }
395
396 template <typename Node, typename Dsp>
397 static void exec(
398 Node& self, Dsp& dsp, const ossia::token_request& tk,
399 const ossia::exec_state_facade& e)
400 {
401 if(tk.forward())
402 {
403 const auto [st, d] = e.timings(tk);
404 copy_controls(self);
405
406 if(d == 0)
407 return;
408 do_exec(self, dsp, tk, e);
409 copy_displays(self, st);
410 }
411 }
412
413 template <typename Node, typename Dsp>
414 static void do_exec_mono_fx(
415 Node& self, Dsp& dsp, const ossia::token_request& tk,
416 const ossia::exec_state_facade& e)
417 {
418 const auto [st, d] = e.timings(tk);
419 if(d == 0)
420 return;
421
422 ossia::audio_port& audio_in
423 = self.root_inputs()[0]->template cast<ossia::audio_port>();
424 ossia::audio_port& audio_out
425 = self.root_outputs()[0]->template cast<ossia::audio_port>();
426
427 const int64_t n_in = audio_in.channels();
428 audio_out.set_channels(n_in);
429 while(self.clones.size() < n_in)
430 {
431 self.clones.emplace_back(dsp.clone(), self.clones[0]);
432 }
433
434 {
435 for(int k = 0; k < self.controls.size(); ++k)
436 {
437 auto ctrl = self.controls[k];
438 auto& dat = ctrl.first->get_data();
439 if(!dat.empty())
440 {
441 if(dat.back().value.valid())
442 ossia::apply_nonnull([k,&self] (const auto& vv){ self.set_control(k, vv); }, dat.back().value.v);
443 }
444 }
445 }
446
447 if constexpr(std::is_same_v<FAUSTFLOAT, float>)
448 {
449 float* input = (float*)alloca(d * sizeof(float));
450 memset(input, 0, d * sizeof(float));
451 float* output = (float*)alloca(d * sizeof(float));
452
453 for(int i = 0; i < n_in; i++)
454 {
455 auto& in_chan = audio_in.channel(i);
456 auto& out_chan = audio_out.channel(i);
457 auto& clone = self.clones[i];
458 in_chan.resize(e.bufferSize());
459 out_chan.resize(e.bufferSize());
460
461 copy_input_mono(self, d, n_in, input, in_chan);
462 memset(output, 0, d * sizeof(float));
463 for(int z = 0; z < d; z++)
464 {
465 assert(!std::isnan(input[z]));
466 assert(!std::isinf(input[z]));
467 }
468 clone.fx->compute(d, &input, &output);
469 for(int z = 0; z < d; z++)
470 {
471 if(std::fpclassify(output[z]) != FP_NORMAL)
472 output[z] = 0.f;
473 }
474
475 std::copy_n(output, d, out_chan.data() + st);
476 for(int z = 0; z < e.bufferSize(); z++)
477 {
478 assert(!std::isnan(out_chan[z]));
479 assert(!std::isinf(out_chan[z]));
480 }
481 }
482 }
483 else
484 {
485 for(int i = 0; i < n_in; i++)
486 {
487 auto& in_chan = audio_in.channel(i);
488 auto& out_chan = audio_out.channel(i);
489 in_chan.resize(e.bufferSize());
490 out_chan.resize(e.bufferSize());
491
492 double* input = in_chan.data() + st;
493 double* output = out_chan.data() + st;
494
495 self.clones[i].fx->compute(d, &input, &output);
496 }
497 }
498 }
499
500 template <typename Node, typename Dsp>
501 static void exec_mono_fx(
502 Node& self, Dsp& dsp, const ossia::token_request& tk,
503 const ossia::exec_state_facade& e)
504 {
505 if(tk.forward())
506 {
507 const auto [st, d] = e.timings(tk);
508
509 if(d == 0)
510 return;
511 do_exec_mono_fx(self, dsp, tk, e);
512 copy_displays(self, st);
513 }
514 }
515
517 template <typename Node, typename DspPoly>
518 void exec_synth(
519 Node& self, DspPoly& dsp, const ossia::token_request& tk,
520 const ossia::exec_state_facade& e)
521 {
522 if(tk.forward())
523 {
524 const auto [st, d] = e.timings(tk);
525
526 auto& midi_in = self.root_inputs()[1]->template cast<ossia::midi_port>();
527
528 copy_controls(self);
529 dsp.updateAllZones();
530 copy_midi(self, dsp, midi_in);
531
532 if(d == 0)
533 return;
534 do_exec(self, dsp, tk, e);
535 copy_displays(self, st);
536 }
537 }
538};
539
540// NOTE: all the code below taken and slightly modified from poly-dsp
541// with small modifications to allow thread-safe update of GroupUI
542// Keep up-to-date with updates in faust.. last update 2020-10-18
543// GPLv3
544// Copyright is the Faust team and Faust contributors
545// Copyright (C) 2003-2017 GRAME, Centre National de Creation Musicale
546class custom_dsp_poly_effect : public dsp_poly
547{
548
549private:
550 mydsp_poly* fPolyDSP;
551
552public:
553 custom_dsp_poly_effect(mydsp_poly* dsp1, dsp* dsp2)
554 : dsp_poly(dsp2)
555 , fPolyDSP(dsp1)
556 {
557 }
558
559 virtual ~custom_dsp_poly_effect()
560 {
561 // dsp_poly_effect is also a decorator_dsp, which will free fPolyDSP
562 }
563
564 void updateAllZones() { fPolyDSP->fGroups.updateAllZones(); }
565 // MIDI API
566 MapUI* keyOn(int channel, int pitch, int velocity)
567 {
568 return fPolyDSP->keyOn(channel, pitch, velocity);
569 }
570 void keyOff(int channel, int pitch, int velocity)
571 {
572 fPolyDSP->keyOff(channel, pitch, velocity);
573 }
574 void keyPress(int channel, int pitch, int press)
575 {
576 fPolyDSP->keyPress(channel, pitch, press);
577 }
578 void chanPress(int channel, int press) { fPolyDSP->chanPress(channel, press); }
579 void ctrlChange(int channel, int ctrl, int value)
580 {
581 fPolyDSP->ctrlChange(channel, ctrl, value);
582 }
583 void ctrlChange14bits(int channel, int ctrl, int value)
584 {
585 fPolyDSP->ctrlChange14bits(channel, ctrl, value);
586 }
587 void pitchWheel(int channel, int wheel) { fPolyDSP->pitchWheel(channel, wheel); }
588 void progChange(int channel, int pgm) { fPolyDSP->progChange(channel, pgm); }
589};
590
591struct custom_dsp_poly_factory : public dsp_factory
592{
593 dsp_factory* fProcessFactory;
594 dsp_factory* fEffectFactory;
595
596 std::vector<std::string> getWarningMessages() { return {}; }
597
598 std::string getEffectCode(const std::string& dsp_content)
599 {
600 std::stringstream effect_code;
601 effect_code << "adapt(1,1) = _; adapt(2,2) = _,_; adapt(1,2) = _ <: _,_; "
602 "adapt(2,1) = _,_ :> _;";
603 effect_code << "adaptor(F,G) = adapt(outputs(F),inputs(G)); dsp_code = "
604 "environment{ "
605 << dsp_content << " };";
606 effect_code << "process = adaptor(dsp_code.process, dsp_code.effect) : "
607 "dsp_code.effect;";
608 return effect_code.str();
609 }
610
611 custom_dsp_poly_factory(
612 dsp_factory* process_factory = NULL, dsp_factory* effect_factory = NULL)
613 : fProcessFactory(process_factory)
614 , fEffectFactory(effect_factory)
615 {
616 }
617
618 virtual ~custom_dsp_poly_factory() = default;
619
620 virtual std::string getName() { return fProcessFactory->getName(); }
621 virtual std::string getSHAKey() { return fProcessFactory->getSHAKey(); }
622 virtual std::string getDSPCode() { return fProcessFactory->getDSPCode(); }
623 virtual std::string getCompileOptions()
624 {
625 return fProcessFactory->getCompileOptions();
626 }
627 virtual std::vector<std::string> getLibraryList()
628 {
629 return fProcessFactory->getLibraryList();
630 }
631 virtual std::vector<std::string> getIncludePathnames()
632 {
633 return fProcessFactory->getIncludePathnames();
634 }
635
636 virtual void setMemoryManager(dsp_memory_manager* manager)
637 {
638 fProcessFactory->setMemoryManager(manager);
639 if(fEffectFactory)
640 {
641 fEffectFactory->setMemoryManager(manager);
642 }
643 }
644 virtual dsp_memory_manager* getMemoryManager()
645 {
646 return fProcessFactory->getMemoryManager();
647 }
648
649 /* Create a new polyphonic DSP instance with global effect, to be deleted
650 * with C++ 'delete'
651 *
652 * @param nvoices - number of polyphony voices, should be at least 1
653 * @param control - whether voices will be dynamically allocated and
654 * controlled (typically by a MIDI controller). If false all voices are always
655 * running.
656 * @param group - if true, voices are not individually accessible, a global
657 * "Voices" tab will automatically dispatch a given control on all voices,
658 * assuming GUI::updateAllGuis() is called. If false, all voices can be
659 * individually controlled.
660 */
661 custom_dsp_poly_effect* createPolyDSPInstance(int nvoices, bool control, bool group)
662 {
663 auto dsp_poly
664 = new mydsp_poly(fProcessFactory->createDSPInstance(), nvoices, control, group);
665 if(fEffectFactory)
666 {
667 // the 'dsp_poly' object has to be controlled with MIDI, so kept
668 // separated from new dsp_sequencer(...) object
669 return new custom_dsp_poly_effect(
670 dsp_poly, new dsp_sequencer(dsp_poly, fEffectFactory->createDSPInstance()));
671 }
672 else
673 {
674 return new custom_dsp_poly_effect(dsp_poly, dsp_poly);
675 }
676 }
677
678 /* Create a new DSP instance, to be deleted with C++ 'delete' */
679 dsp* createDSPInstance() { return fProcessFactory->createDSPInstance(); }
680};
681struct custom_llvm_dsp_poly_factory : public custom_dsp_poly_factory
682{
683 custom_llvm_dsp_poly_factory(
684 const std::string& name_app, const std::string& dsp_content, int argc,
685 const char* argv[], const std::string& target, std::string& error_msg,
686 int opt_level = -1)
687 {
688 fProcessFactory = createDSPFactoryFromString(
689 name_app, dsp_content, argc, argv, target, error_msg);
690 if(fProcessFactory)
691 {
692 fEffectFactory = createDSPFactoryFromString(
693 name_app, getEffectCode(dsp_content), argc, argv, target, error_msg);
694 if(!fEffectFactory)
695 {
696 std::cerr << "llvm_dsp_poly_factory : fEffectFactory " << error_msg;
697 // The error message is not really needed...
698 error_msg = "";
699 }
700 }
701 else
702 {
703 std::cerr << "llvm_dsp_poly_factory : fProcessFactory " << error_msg;
704 throw std::bad_alloc();
705 }
706 }
707
708 virtual ~custom_llvm_dsp_poly_factory()
709 {
710 deleteDSPFactory(static_cast<llvm_dsp_factory*>(fProcessFactory));
711 deleteDSPFactory(static_cast<llvm_dsp_factory*>(fEffectFactory));
712 }
713};
714
715static custom_llvm_dsp_poly_factory* createCustomPolyDSPFactoryFromString(
716 const std::string& name_app, const std::string& dsp_content, int argc,
717 const char* argv[], const std::string& target, std::string& error_msg,
718 int opt_level = -1)
719{
720 try
721 {
722 return new custom_llvm_dsp_poly_factory(
723 name_app, dsp_content, argc, argv, target, error_msg, opt_level);
724 }
725 catch(...)
726 {
727 return NULL;
728 }
729}
730
731}
OSSIA_INLINE constexpr auto min(const T a, const U b) noexcept -> typename std::conditional<(sizeof(T) > sizeof(U)), T, U >::type
min function tailored for values
Definition math.hpp:125
OSSIA_INLINE constexpr auto max(const T a, const U b) noexcept -> typename std::conditional<(sizeof(T) > sizeof(U)), T, U >::type
max function tailored for values
Definition math.hpp:96