227 lines
11 KiB
Scheme
227 lines
11 KiB
Scheme
|
|
(define type->native-type ; Chicken has this procedure in three places
|
|
(lambda (type)
|
|
(cond ((equal? type 'int8) 'byte)
|
|
((equal? type 'uint8) 'unsigned-byte)
|
|
((equal? type 'int16) 'short)
|
|
((equal? type 'uint16) 'unsigned-short)
|
|
((equal? type 'int32) 'integer32)
|
|
((equal? type 'uint32) 'unsigned-integer32)
|
|
((equal? type 'int64) 'integer64)
|
|
((equal? type 'uint64) 'unsigned-integer64)
|
|
((equal? type 'char) 'char)
|
|
((equal? type 'unsigned-char) 'unsigned-char)
|
|
((equal? type 'short) 'short)
|
|
((equal? type 'unsigned-short) 'unsigned-short)
|
|
((equal? type 'int) 'int)
|
|
((equal? type 'unsigned-int) 'unsigned-int)
|
|
((equal? type 'long) 'long)
|
|
((equal? type 'unsigned-long) 'unsigned-long)
|
|
((equal? type 'float) 'float)
|
|
((equal? type 'double) 'double)
|
|
((equal? type 'pointer) 'c-pointer)
|
|
((equal? type 'void) 'void)
|
|
((equal? type 'callback) 'c-pointer)
|
|
((equal? type 'struct) 'c-pointer)
|
|
(else (error "type->native-type -- No such pffi type" type)))))
|
|
|
|
(define c-bytevector?
|
|
(lambda (object)
|
|
(pointer? object)))
|
|
|
|
(define-syntax define-c-procedure
|
|
(er-macro-transformer
|
|
(lambda (expr rename compare)
|
|
(let* ((type->native-type ; Chicken has this procedure in three places
|
|
(lambda (type)
|
|
(cond ((equal? type 'int8) 'byte)
|
|
((equal? type 'uint8) 'unsigned-byte)
|
|
((equal? type 'int16) 'short)
|
|
((equal? type 'uint16) 'unsigned-short)
|
|
((equal? type 'int32) 'integer32)
|
|
((equal? type 'uint32) 'unsigned-integer32)
|
|
((equal? type 'int64) 'integer64)
|
|
((equal? type 'uint64) 'unsigned-integer64)
|
|
((equal? type 'char) 'char)
|
|
((equal? type 'unsigned-char) 'unsigned-char)
|
|
((equal? type 'short) 'short)
|
|
((equal? type 'unsigned-short) 'unsigned-short)
|
|
((equal? type 'int) 'int)
|
|
((equal? type 'unsigned-int) 'unsigned-int)
|
|
((equal? type 'long) 'long)
|
|
((equal? type 'unsigned-long) 'unsigned-long)
|
|
((equal? type 'float) 'float)
|
|
((equal? type 'double) 'double)
|
|
((equal? type 'pointer) 'c-pointer)
|
|
((equal? type 'void) 'void)
|
|
((equal? type 'callback) 'c-pointer)
|
|
((equal? type 'struct) 'c-pointer)
|
|
(else (error "type->native-type -- No such pffi type" type)))))
|
|
(scheme-name (list-ref expr 1))
|
|
(c-name (symbol->string (cadr (list-ref expr 3))))
|
|
(return-type (type->native-type (cadr (list-ref expr 4))))
|
|
(argument-types (if (null? (cdr (list-ref expr 5)))
|
|
(list)
|
|
(map type->native-type
|
|
(cadr (list-ref expr 5))))))
|
|
(if (null? argument-types)
|
|
`(define ,scheme-name
|
|
(foreign-safe-lambda ,return-type ,c-name))
|
|
`(define ,scheme-name
|
|
(foreign-safe-lambda ,return-type ,c-name ,@ argument-types)))))))
|
|
|
|
(define-syntax define-c-callback
|
|
(er-macro-transformer
|
|
(lambda (expr rename compare)
|
|
(let* ((type->native-type ; Chicken has this procedure in three places
|
|
(lambda (type)
|
|
(cond ((equal? type 'int8) 'byte)
|
|
((equal? type 'uint8) 'unsigned-byte)
|
|
((equal? type 'int16) 'short)
|
|
((equal? type 'uint16) 'unsigned-short)
|
|
((equal? type 'int32) 'integer32)
|
|
((equal? type 'uint32) 'unsigned-integer32)
|
|
((equal? type 'int64) 'integer64)
|
|
((equal? type 'uint64) 'unsigned-integer64)
|
|
((equal? type 'char) 'char)
|
|
((equal? type 'unsigned-char) 'unsigned-char)
|
|
((equal? type 'short) 'short)
|
|
((equal? type 'unsigned-short) 'unsigned-short)
|
|
((equal? type 'int) 'int)
|
|
((equal? type 'unsigned-int) 'unsigned-int)
|
|
((equal? type 'long) 'long)
|
|
((equal? type 'unsigned-long) 'unsigned-long)
|
|
((equal? type 'float) 'float)
|
|
((equal? type 'double) 'double)
|
|
((equal? type 'pointer) 'c-pointer)
|
|
((equal? type 'void) 'void)
|
|
((equal? type 'callback) 'c-pointer)
|
|
((equal? type 'struct) 'c-pointer)
|
|
(else (error "type->native-type -- No such pffi type" type)))))
|
|
(scheme-name (list-ref expr 1))
|
|
(return-type (type->native-type (cadr (list-ref expr 2))))
|
|
(argument-types (map type->native-type (cadr (list-ref expr 3))))
|
|
(argument-names (cadr (list-ref expr 4)))
|
|
(arguments (map
|
|
(lambda (name type)
|
|
`(,name ,type))
|
|
argument-types argument-names))
|
|
(procedure-body (cdr (cdr (list-ref expr 4)))))
|
|
`(begin (define-external ,(cons 'external_123456789 arguments)
|
|
,return-type
|
|
(begin ,@ procedure-body))
|
|
(define ,scheme-name (location external_123456789)))))))
|
|
|
|
(define size-of-type
|
|
(lambda (type)
|
|
(cond ((equal? type 'int8) (foreign-value "sizeof(int8_t)" int))
|
|
((equal? type 'uint8) (foreign-value "sizeof(uint8_t)" int))
|
|
((equal? type 'int16) (foreign-value "sizeof(int16_t)" int))
|
|
((equal? type 'uint16) (foreign-value "sizeof(uint16_t)" int))
|
|
((equal? type 'int32) (foreign-value "sizeof(int32_t)" int))
|
|
((equal? type 'uint32) (foreign-value "sizeof(uint32_t)" int))
|
|
((equal? type 'int64) (foreign-value "sizeof(int64_t)" int))
|
|
((equal? type 'uint64) (foreign-value "sizeof(uint64_t)" int))
|
|
((equal? type 'char) (foreign-value "sizeof(char)" int))
|
|
((equal? type 'unsigned-char) (foreign-value "sizeof(unsigned char)" int))
|
|
((equal? type 'short) (foreign-value "sizeof(short)" int))
|
|
((equal? type 'unsigned-short) (foreign-value "sizeof(unsigned short)" int))
|
|
((equal? type 'int) (foreign-value "sizeof(int)" int))
|
|
((equal? type 'unsigned-int) (foreign-value "sizeof(unsigned int)" int))
|
|
((equal? type 'long) (foreign-value "sizeof(long)" int))
|
|
((equal? type 'unsigned-long) (foreign-value "sizeof(unsigned long)" int))
|
|
((equal? type 'float) (foreign-value "sizeof(float)" int))
|
|
((equal? type 'double) (foreign-value "sizeof(double)" int))
|
|
((equal? type 'pointer) (foreign-value "sizeof(void*)" int))
|
|
((equal? type 'string) (foreign-value "sizeof(void*)" int))
|
|
((equal? type 'callback) (foreign-value "sizeof(void*)" int)))))
|
|
|
|
(define make-c-null
|
|
(lambda ()
|
|
(address->pointer 0)))
|
|
|
|
(define-syntax define-c-library
|
|
(syntax-rules ()
|
|
((_ scheme-name headers object-name options)
|
|
(begin
|
|
(define scheme-name #t)
|
|
(shared-object-load headers)))))
|
|
|
|
(define-syntax shared-object-load
|
|
(er-macro-transformer
|
|
(lambda (expr rename compare)
|
|
(let* ((headers (cadr (car (cdr expr)))))
|
|
`(begin
|
|
,@ (map
|
|
(lambda (header)
|
|
`(foreign-declare ,(string-append "#include <" header ">")))
|
|
headers))))))
|
|
|
|
(define c-null?
|
|
(lambda (pointer)
|
|
(if (and (not (pointer? pointer))
|
|
pointer)
|
|
#f
|
|
(or (not pointer) ; #f counts as null pointer on Chicken
|
|
(= (pointer->address pointer) 0)))))
|
|
|
|
(define c-bytevector-u8-ref
|
|
(lambda (c-bytevector k)
|
|
(pointer-u8-ref (pointer+ c-bytevector k))))
|
|
|
|
(define c-bytevector-u8-set!
|
|
(lambda (c-bytevector k byte)
|
|
(pointer-u8-set! (pointer+ c-bytevector k) byte)))
|
|
|
|
(define c-bytevector-pointer-ref
|
|
(lambda (c-bytevector k)
|
|
(address->pointer (pointer-u64-ref (pointer+ c-bytevector k)))))
|
|
|
|
(define c-bytevector-pointer-set!
|
|
(lambda (c-bytevector k pointer)
|
|
(pointer-u64-set! (pointer+ c-bytevector k) (pointer->address pointer))))
|
|
|
|
#;(define pffi-pointer-set!
|
|
(lambda (pointer type offset value)
|
|
(cond
|
|
((equal? type 'int8) (pointer-s8-set! (pointer+ pointer offset) value))
|
|
((equal? type 'uint8) (pointer-u8-set! (pointer+ pointer offset) value))
|
|
((equal? type 'int16) (pointer-s16-set! (pointer+ pointer offset) value))
|
|
((equal? type 'uint16) (pointer-u16-set! (pointer+ pointer offset) value))
|
|
((equal? type 'int32) (pointer-s32-set! (pointer+ pointer offset) value))
|
|
((equal? type 'uint32) (pointer-u32-set! (pointer+ pointer offset) value))
|
|
((equal? type 'int64) (pointer-s64-set! (pointer+ pointer offset) value))
|
|
((equal? type 'uint64) (pointer-u64-set! (pointer+ pointer offset) value))
|
|
((equal? type 'char) (pointer-s8-set! (pointer+ pointer offset) (char->integer value)))
|
|
((equal? type 'short) (pointer-s8-set! (pointer+ pointer offset) value))
|
|
((equal? type 'unsigned-short) (pointer-u8-set! (pointer+ pointer offset) value))
|
|
((equal? type 'int) (pointer-s32-set! (pointer+ pointer offset) value))
|
|
((equal? type 'unsigned-int) (pointer-u32-set! (pointer+ pointer offset) value))
|
|
((equal? type 'long) (pointer-s32-set! (pointer+ pointer offset) value))
|
|
((equal? type 'unsigned-long) (pointer-u32-set! (pointer+ pointer offset) value))
|
|
((equal? type 'float) (pointer-f32-set! (pointer+ pointer offset) value))
|
|
((equal? type 'double) (pointer-f64-set! (pointer+ pointer offset) value))
|
|
((equal? type 'pointer) (pointer-u64-set! (pointer+ pointer offset) (pointer->address value))))))
|
|
|
|
#;(define pffi-pointer-get
|
|
(lambda (pointer type offset)
|
|
(cond
|
|
((equal? type 'int8) (pointer-s8-ref (pointer+ pointer offset)))
|
|
((equal? type 'uint8) (pointer-u8-ref (pointer+ pointer offset)))
|
|
((equal? type 'int16) (pointer-s16-ref (pointer+ pointer offset)))
|
|
((equal? type 'uint16) (pointer-u16-ref (pointer+ pointer offset)))
|
|
((equal? type 'int32) (pointer-s32-ref (pointer+ pointer offset)))
|
|
((equal? type 'uint32) (pointer-u32-ref (pointer+ pointer offset)))
|
|
((equal? type 'int64) (pointer-s64-ref (pointer+ pointer offset)))
|
|
((equal? type 'uint64) (pointer-u64-ref (pointer+ pointer offset)))
|
|
((equal? type 'char) (integer->char (pointer-s8-ref (pointer+ pointer offset))))
|
|
((equal? type 'short) (pointer-s8-ref (pointer+ pointer offset)))
|
|
((equal? type 'unsigned-short) (pointer-u8-ref (pointer+ pointer offset)))
|
|
((equal? type 'int) (pointer-s32-ref (pointer+ pointer offset)))
|
|
((equal? type 'unsigned-int) (pointer-u32-ref (pointer+ pointer offset)))
|
|
((equal? type 'long) (pointer-s32-ref (pointer+ pointer offset)))
|
|
((equal? type 'unsigned-long) (pointer-u32-ref (pointer+ pointer offset)))
|
|
((equal? type 'float) (pointer-f32-ref (pointer+ pointer offset)))
|
|
((equal? type 'double) (pointer-f64-ref (pointer+ pointer offset)))
|
|
((equal? type 'pointer) (address->pointer (pointer-u64-ref (pointer+ pointer offset)))))))
|