{- | Implementation of server b_gen routines. The naming scheme is: _p generates one partial, _l generates a list of partials, _nrm is the unit normalised form. -} module Sound.Sc3.Common.Buffer.Gen where import Data.List {- base -} import qualified Sound.Sc3.Common.Buffer as Buffer {- hsc3 -} import qualified Sound.Sc3.Common.Math as Math {- hsc3 -} -- | Sum (mix) multiple tables into one. sum_l :: Num n => [[n]] -> [n] sum_l = map sum . transpose -- | Unit normalisation. nrm_u :: (Fractional n, Ord n) => [n] -> [n] nrm_u = Buffer.normalize (-1) 1 -- * sine1 {- | 'sine3_p' with zero phase. > import Sound.Sc3.Plot > plot_p1_ln [sine1_p 512 (1, 1)] -} sine1_p :: (Enum n, Floating n) => Int -> (n, n) -> [n] sine1_p n (pfreq, ampl) = sine3_p n (pfreq, ampl, 0) -- | Series of sine wave harmonics using specified amplitudes. sine1_l :: (Enum n, Floating n) => Int -> [n] -> [[n]] sine1_l n = zipWith (curry (sine1_p n)) [1 ..] {- | 'sum_l' of 'sine1_l'. > plot_p1_ln [sine1 256 [1, 0.95 .. 0.5]] -} sine1 :: (Enum n, Floating n) => Int -> [n] -> [n] sine1 n = sum_l . sine1_l n {- | 'nrm_u' of 'sine1_l'. > Sound.Sc3.Plot.plot_p1_ln [sine1_nrm 256 [1, 0.95 .. 0.5]] > Sound.Sc3.Plot.plot_p1_ln [sine1_nrm 256 [1, 1/2, 1/3, 1/4, 1/5]] -} sine1_nrm :: (Enum n, Floating n, Ord n) => Int -> [n] -> [n] sine1_nrm n = nrm_u . sine1 n -- | Variant that generates a wavetable (without guard point) suitable for the Shaper Ugen. sine1Tbl :: (Enum n, Floating n, Ord n) => Int -> [n] -> [n] sine1Tbl n = Buffer.to_wavetable . sine1_nrm n -- * sine2 -- | Series of /n/ sine wave partials using specified frequencies and amplitudes. sine2_l :: (Enum n, Floating n) => Int -> [(n, n)] -> [[n]] sine2_l n = map (sine1_p n) {- | 'sum_l' of 'sine2_l'. > Sound.Sc3.Plot.plot_p1_ln [sine2 256 (zip [1, 2..] [1, 0.95 .. 0.5])] > Sound.Sc3.Plot.plot_p1_ln [sine2 256 (zip [1, 1.5 ..] [1, 0.95 .. 0.5])] -} sine2 :: (Enum n, Floating n) => Int -> [(n, n)] -> [n] sine2 n = sum_l . sine2_l n -- | 'nrm_u' of 'sine2_l'. sine2_nrm :: (Enum n, Floating n, Ord n) => Int -> [n] -> [n] sine2_nrm n = nrm_u . sine1 n -- * sine3 {- | Sine wave table at specified frequency, amplitude and phase. The table does not arrive back at the starting point. >>> map (round . (* 100)) (sine3_p 8 (1, 1, 0)) [0,71,100,71,0,-71,-100,-71] -} sine3_p :: (Enum n, Floating n) => Int -> (n, n, n) -> [n] sine3_p n (pfreq, ampl, phase) = let incr = (Math.two_pi / (fromIntegral n - 0)) * pfreq in map ((*) ampl . sin) (take n [phase, phase + incr ..]) -- | 'map' of 'sine3_p'. sine3_l :: (Enum n, Floating n) => Int -> [(n, n, n)] -> [[n]] sine3_l n = map (sine3_p n) {- | 'sum_l' of 'sine3_l'. > plot_p1_ln [sine3 256 (zip3 [1,1.5 ..] [1,0.95 .. 0.5] [0,pi/7..])] -} sine3 :: (Enum n, Floating n) => Int -> [(n, n, n)] -> [n] sine3 n = sum_l . sine3_l n -- * cheby {- | Generate Chebyshev waveshaping table, see b_gen_cheby. Cf. > import Sound.Sc3.Plot > let p x = plot_p1_ln [gen_cheby 512 x] > p [1, 0, 1, 1, 0, 1] > p [100, -50, -33, 25, 20, -16.7, -14.2, 12.5, 11.1, -10, -9.09, 8.333, 7.69, -7.14, -6.67, 6.25, 5.88, -5.55, -5.26, 5] > p [100, 0, -33, 0, 20, 0, -14.2, 0, 11.1, 0, -9.09, 0, 7.69, 0, -6.67, 0, 5.88, 0, -5.26] > p [100, 0, -11.11, 0, 4, 0, -2.04, 0, 1.23, 0, -0.826, 0, 0.59, 0, -0.444, 0, 0.346, 0, -0.277] > p [1, -0.8, 0, 0.6, 0, 0, 0, 0.4, 0, 0, 0, 0, 0.1, -0.2, -0.3, 0.5] > p [0, 0, -0.1, 0, 0.3, 0, -0.5, 0, 0.7, 0, -0.9, 0, 1, 0, -1, 0] > p [0, 0, 0, 0, 0, 0, -1, 0, 1, 0, 0, -0.1, 0, 0.1, 0, -0.2, 0.3, 0, -0.7, 0, 0.2, 0, -0.1] > p [5, 0, 3, 0, 1] -} gen_cheby :: (Enum n, Floating n, Ord n, Integral i) => i -> [n] -> [n] gen_cheby n = let acos' x = if x > 1 then 0 else if x < -1 then pi else acos x c k x = cos (k * acos' x) ix = [-1, -1 + (2 / (fromIntegral n - 1)) .. 1] -- increment? mix = map sum . transpose c_normalize x = let m = maximum (map abs x) in map (* recip m) x in c_normalize . mix . zipWith (\k a -> map ((* a) . c k) ix) [1 ..] -- | Type specialised 'gen_cheby'. cheby :: (Enum n, Floating n, Ord n) => Int -> [n] -> [n] cheby = gen_cheby -- | Variant that generates a wavetable (without guard point) suitable for the Shaper Ugen. chebyShaperTbl :: (Enum n, Floating n, Ord n) => Int -> [n] -> [n] chebyShaperTbl n = Buffer.to_wavetable_nowrap . cheby n