factor/libs/mad/api.factor

103 lines
3.7 KiB
Factor

! Copyright (C) 2007 Adam Wendt.
! See http://factorcode.org/license.txt for BSD license.
!
IN: mad
USING: kernel alien alien-contrib namespaces threads errors
sequences shuffle compiler prettyprint io tools interpreter vars math sbufs byte-arrays ;
VARS: buffer-start buffer-length ;
: create-mad-callback-generic ( sequence parameters -- alien )
swap >r >r "mad_flow" r> "cdecl" r> alien-callback ; inline
: create-input-callback ( sequence -- alien )
{ "void*" "mad_stream*" } create-mad-callback-generic ; inline
: create-header-callback ( sequence -- alien )
{ "void*" "mad_header*" } create-mad-callback-generic ; inline
: create-filter-callback ( sequence -- alien )
{ "void*" "mad_stream*" "mad_frame*" } create-mad-callback-generic ; inline
: create-output-callback ( sequence -- alien )
{ "void*" "mad_header*" "mad_pcm*" } create-mad-callback-generic ; inline
: create-error-callback ( sequence -- alien )
{ "void*" "mad_stream*" "mad_frame*" } create-mad-callback-generic ; inline
: create-message-callback ( sequence -- alien )
{ "void*" "void*" "uint*" } create-mad-callback-generic ; inline
: input ( buffer mad_stream -- mad_flow )
! "input" print flush
nip ! mad_stream
buffer-start get ! mad_stream start
buffer-length get ! mad_stream start length
dup 0 = ! mad-stream start length bool
[ 3drop MAD_FLOW_STOP ] ! mad_flow
[ mad_stream_buffer !
0 buffer-length set !
MAD_FLOW_CONTINUE ] if ; ! mad_flow
: input-callback ( -- callback )
[ input ] create-input-callback ;
: header-callback ( -- callback )
[ "header" print flush drop drop MAD_FLOW_CONTINUE ] create-header-callback ;
: filter-callback ( -- callback )
[ "filter" print flush 3drop MAD_FLOW_CONTINUE ] create-filter-callback ;
: output-sample ( sample -- )
! dup 0 shift HEX: ff bitand write1 -8 shift HEX: ff bitand write1 ;
4 >le write ;
: output ( data header pcm -- mad_flow )
! "output" print flush
-rot 2drop ! pcm
[ mad_pcm-channels ] keep ! channels pcm
[ mad_pcm-length ] keep swap ! channels pcm nsamples
[ ! channels pcm counter
[ mad_pcm-sample-right ] 2keep ! channels right pcm counter
[ mad_pcm-sample-left ] 2keep ! channels right left pcm counter
drop -rot ! channels pcm right left
output-sample pick ! channels pcm right channels
2 = [ output-sample ] [ drop ] if ! channels
] each ! channels
drop MAD_FLOW_CONTINUE ;
: output-callback ( -- callback )
[ output ] create-output-callback ;
: error-callback ( -- callback )
[ "error" print flush drop drop drop MAD_FLOW_CONTINUE ] create-error-callback ;
: message-callback ( -- callback )
[ "message" print flush drop drop drop MAD_FLOW_CONTINUE ] create-message-callback ;
: mad-init ( decoder -- )
0 <alien> input-callback 0 <alien> 0 <alien> output-callback error-callback message-callback mad_decoder_init ;
: make-decoder ( -- decoder )
"mad_decoder" malloc-object ;
: file-contents ( path -- string )
dup <file-reader> swap file-length <sbuf> [ stream-copy ] keep >byte-array ;
: malloc-file-contents ( path -- alien )
file-contents dup length malloc-byte-array ;
: mad-run ( -- int )
make-decoder [ mad-init ] keep MAD_DECODER_MODE_SYNC mad_decoder_run ;
: init-vars ( alien length -- )
buffer-length set buffer-start set ;
: play-mp3 ( filename -- results )
[ malloc-file-contents ] keep file-length init-vars mad-run ;
: mad-test ( -- results )
"/home/adam/download/mp3/Misc/wutbf.mp3" play-mp3 ;