From 867496f6bb4703815a5854e1ff11f795c49e3702 Mon Sep 17 00:00:00 2001 From: Johannes Waldmann Date: Sun, 28 Jan 2024 22:12:05 +0100 Subject: [PATCH] for #1068 --- src/Sound/Tidal/UI.hs | 22 +++++++++++++++++++++- tidal.cabal | 1 + 2 files changed, 22 insertions(+), 1 deletion(-) diff --git a/src/Sound/Tidal/UI.hs b/src/Sound/Tidal/UI.hs index 21b62e03d..d063f3c41 100644 --- a/src/Sound/Tidal/UI.hs +++ b/src/Sound/Tidal/UI.hs @@ -1,4 +1,4 @@ -{-# LANGUAGE TypeSynonymInstances, FlexibleInstances, OverloadedStrings #-} +{-# LANGUAGE TypeSynonymInstances, FlexibleInstances, OverloadedStrings, BangPatterns #-} {- UI.hs - Tidal's main 'user interface' functions, for transforming @@ -45,6 +45,10 @@ import Data.Maybe (isJust, fromJust, fromMaybe, mapMaybe) import qualified Data.Text as T import qualified Data.Map.Strict as Map import Data.Bool (bool) +import qualified Data.List as L +import qualified Data.Vector as V +import qualified Data.Vector.Unboxed as U + import Sound.Tidal.Bjorklund (bjorklund) import Sound.Tidal.Core @@ -1226,12 +1230,28 @@ runMarkov 8 [[2,3], [1,3]] 0 0 will produce a two-state chain 8 steps long, from initial state @0@, where the transition probability from state 0->0 is 2/5, 0->1 is 3/5, 1->0 is 1/4, and 1->1 is 3/4. -} + runMarkov :: Int -> [[Double]] -> Int -> Time -> [Int] runMarkov n tp xi seed = reverse $ (iterate (markovStep $ renorm) [xi])!! (n-1) where markovStep tp' xs = (fromJust $ findIndex (r <=) $ scanl1 (+) (tp'!!(head xs))) : xs where r = timeToRand $ seed + (fromIntegral . length) xs / fromIntegral n renorm = [ map (/ sum x) x | x <- tp ] +runMarkov' :: Int -> [[Double]] -> Int -> Time -> [Int] +runMarkov' n tp xi seed = take n $ map fst $ L.iterate' (markovStep $ renorm) (xi, seed + delta) where + markovStep tp' (x,seed) = (let (s,v) = tp' V.! x in binarySearch 0 (r * s) v , seed + delta) where + r = timeToRand seed + renorm :: V.Vector (Double, U.Vector Double) + renorm = V.fromList [ fmap U.fromList $ L.mapAccumL (\ a y -> let s = a+y in s `seq` (s,s)) 0 x | x <- tp ] + binarySearch :: Int -> Double -> U.Vector Double -> Int + binarySearch !off x v = + if U.length v == 0 then off + else if U.length v == 1 then off + 1 + else let i = div (U.length v) 2 + in if x < v U.! i then binarySearch off x $ U.slice 0 i v + else binarySearch (off + i) x (U.slice i (U.length v - i) v) + delta = 1 / fromIntegral n + {- @markovPat n xi tp@ generates a one-cycle pattern of @n@ steps in a Markov chain starting from state @xi@ with transition matrix @tp@. Each row of the transition matrix is automatically normalized. For example: diff --git a/tidal.cabal b/tidal.cabal index d242b7eb7..9ff7e9e1f 100644 --- a/tidal.cabal +++ b/tidal.cabal @@ -67,6 +67,7 @@ library , random < 1.3 , exceptions < 0.11 , mtl >= 2.2 + , vector , tidal-link == 1.0.2 test-suite tests