mirror of
https://github.com/TomHarte/CLK.git
synced 2025-01-24 02:30:54 +00:00
0aceddd088
This is as a precursor to switching to using the proper table lookups, which I hope will automatically fix my range issues.
82 lines
2.6 KiB
C++
82 lines
2.6 KiB
C++
//
|
|
// Channel.hpp
|
|
// Clock Signal
|
|
//
|
|
// Created by Thomas Harte on 15/04/2020.
|
|
// Copyright © 2020 Thomas Harte. All rights reserved.
|
|
//
|
|
|
|
#ifndef Channel_hpp
|
|
#define Channel_hpp
|
|
|
|
#include "Operator.hpp"
|
|
|
|
#include <cmath>
|
|
|
|
namespace Yamaha {
|
|
namespace OPL {
|
|
|
|
/*!
|
|
Models an L-type two-operator channel.
|
|
|
|
Assuming FM synthesis is enabled, the channel modulates the output of the carrier with that of the modulator.
|
|
|
|
TODO: make this a template on period counter size in bits?
|
|
*/
|
|
class Channel {
|
|
public:
|
|
/// Sets the low 8 bits of frequency control.
|
|
void set_frequency_low(uint8_t value);
|
|
|
|
/// Sets the high two bits of a 10-bit frequency control, along with this channel's
|
|
/// block/octave, and key on or off.
|
|
void set_10bit_frequency_octave_key_on(uint8_t value);
|
|
|
|
/// Sets the high two bits of a 9-bit frequency control, along with this channel's
|
|
/// block/octave, and key on or off.
|
|
void set_9bit_frequency_octave_key_on(uint8_t value);
|
|
|
|
/// Sets the amount of feedback provided to the first operator (i.e. the modulator)
|
|
/// associated with this channel, and whether FM synthesis is in use.
|
|
void set_feedback_mode(uint8_t value);
|
|
|
|
/// This should be called at a rate of around 49,716 Hz; it returns the current output level
|
|
/// level for this channel.
|
|
int update(Operator *modulator, Operator *carrier, OperatorOverrides *modulator_overrides = nullptr, OperatorOverrides *carrier_overrides = nullptr);
|
|
|
|
/// @returns @c true if this channel is currently producing any audio; @c false otherwise;
|
|
bool is_audible(Operator *carrier, OperatorOverrides *carrier_overrides = nullptr);
|
|
|
|
private:
|
|
float level(OperatorState &state, float modulator_level);
|
|
|
|
/// 'F-Num' in the spec; this plus the current octave determines channel frequency.
|
|
int period_ = 0;
|
|
|
|
/// Linked with the frequency, determines the channel frequency.
|
|
int octave_ = 0;
|
|
|
|
/// Sets sets this channel on or off, as an input to the ADSR envelope,
|
|
bool key_on_ = false;
|
|
|
|
/// Sets the degree of feedback applied to the modulator.
|
|
int feedback_strength_ = 0;
|
|
|
|
/// Selects between FM synthesis, using the modulator to modulate the carrier, or simple mixing of the two
|
|
/// underlying operators as completely disjoint entities.
|
|
bool use_fm_synthesis_ = true;
|
|
|
|
/// Used internally to make both the 10-bit OPL2 frequency selection and 9-bit OPLL/VRC7 frequency
|
|
/// selections look the same when passed to the operators.
|
|
int frequency_shift_ = 0;
|
|
|
|
// Stored separately because carrier/modulator may not be unique per channel —
|
|
// on the OPLL there's an extra level of indirection.
|
|
OperatorState carrier_state_, modulator_state_;
|
|
};
|
|
|
|
}
|
|
}
|
|
|
|
#endif /* Channel_hpp */
|