将CSound嵌入Common Lisp

时间:2017-05-18 12:46:36

标签: common-lisp csound

我正在努力将CSound嵌入到Lisp中。 CSound is a music synthesis (and more) open source software.

它有一种相当简单的(脚本)语言。上面的链接提供快速启动(10分钟读取)。目前我正在研究任务部分(这是csound语言的很大一部分)。

这是我的代码:

(defparameter *assign-statements* nil)

(defmacro assign (_name value &optional (rate 'i))
  (let* ((name (if (typep _name 'symbol) _name (eval _name)))
         (var (symb (format nil "~(~a~)" rate) name)))
    `(progn 
       (defparameter ,name ',var)
       (defparameter ,var ,value)
       (setf *assign-statements* 
                 (cons (format nil "~A = ~A" ,name ,value) *assign-statements*)))))

(defmacro assign* (&rest args)
  `(progn ,@(mapcar (lambda (arg) (cons 'assign arg)) args)))

(defun opcode-call (opcode &rest args)
  (format nil "~a ~{~a~^, ~}" opcode (mapcar (lambda (arg) 
             (if (stringp arg) 
                 (let ((var (gensym)))
                   (eval (list 'assign (symb (symbol-name var)) arg 'a))
                   (symbol-value (symb (symbol-name var)))) 
                 arg)) 
          args)))

(defmacro op (opcode &rest args)
  `(opcode-call ',opcode ,@args))

演示代码的作用:

(progn  
  (defparameter *assign-statements* nil)
  (assign* 
    (freq 'p4)
    (amp 'p5)
    (att (+ 0.1 0.1))
    (dec 0.4)
    (sus 0.6)
    (rel 0.7)
    (cutoff 5000)
    (res 0.4 k)
    (env (op madsr (op moogladder freq amp) att dec sus rel) k))
  (format t "~{~A~^~%~}~%" 
      (nreverse *assign-statements*)))

输出:

iFREQ = P4
iAMP = P5
iATT = 0.2
iDEC = 0.4
iSUS = 0.6
iREL = 0.7
iCUTOFF = 5000
kRES = 0.4
aG8707 = MOOGLADDER iFREQ, iAMP
aG8708 = MOOGLADDER iFREQ, iAMP
kENV = MADSR aG8708, iATT, iDEC, iSUS, iREL
NIL  

这在所有方面都是正确的,除了“MOOGLADDER iFREQ,iAMP”出现两次。

这是为什么?我无法弄清楚它被评估两次的位置。 如何消除这种重复?

关于代码的说明:

  • Csound具有a,k和i速率变量的概念。这是 奇怪地实现为变量符号的前缀。最近的 lisp中的对应物将是全局变量的对应物。所以我 这样实现了它。但是为了适应这个比率,我有一个 符号与其值之间的间接级别。例如该 符号'res为其价值'kRes。现在符号'kRes有它的 价值,原来的0.4。

  • 宏'op和'assign *分别是'opcode-call和'assign的简单包装。

  • 'opcode-call是一个函数,因此自动允许正常 订单评估,从而允许嵌套函数调用,其中 csound本身没有(完全)支持。为了解决这个问题,'opcode-call通过检查它的类型在其参数列表中查找任何已评估的操作码调用 (串)。如果它找到一个字符串,它将用gensym替换它 变量

  • 每次分配调用都会将分配添加到assign列表中 语句,然后最终用于输出到csound语言。

1 个答案:

答案 0 :(得分:4)

您的宏ASSIGN允许值计算两次。请参阅下面的评论。

(defmacro assign (_name value &optional (rate 'i))
  (let* ((name (if (typep _name 'symbol) _name (eval _name)))
         (var (symb (format nil "~(~a~)" rate) name)))
    `(progn 
       (defparameter ,name ',var)
       (defparameter ,var ,value)
       (push (format nil "~A = ~A" ,name ,var)    ; <- use the var
             *assign-statements*))))

试一试:

CL-USER 52 > (progn  
               (defparameter *assign-statements* nil)
               (assign* 
                (freq 'p4)
                (amp 'p5)
                (att (+ 0.1 0.1))
                (dec 0.4)
                (sus 0.6)
                (rel 0.7)
                (cutoff 5000)
                (res 0.4 k)
                (env (op madsr (op moogladder freq amp) att dec sus rel) k))
               (format t "~{~A~^~%~}~%" 
                       (nreverse *assign-statements*)))
iFREQ = P4
iAMP = P5
iATT = 0.2
iDEC = 0.4
iSUS = 0.6
iREL = 0.7
iCUTOFF = 5000
kRES = 0.4
aG2719 = MOOGLADDER iFREQ, iAMP
kENV = MADSR aG2719, iATT, iDEC, iSUS, iREL
NIL