(module libcrypto mzscheme
(require-for-syntax "stx-util.ss")
(require (lib "foreign.ss"))
(require "compat.ss" (only "stx-util.ss" ->string))
(unsafe!)
(provide (all-defined)
(all-from "compat.ss"))
(define libcrypto
(case (system-type)
((windows) (ffi-lib "libeay32"))
(else (ffi-lib "libcrypto"))))
(define *silent* #f)
(define (make-unavailable sym)
(lambda x (error sym "unavailable foreign function")))
(define (make-failure-thunk sym)
(lambda ()
(unless *silent*
(fprintf (current-error-port)
"warning: unavailable foreign function: ~a~n" sym))
(make-unavailable sym)))
(define (ffi-available? sym)
(and (get-ffi-obj (->string sym) libcrypto _pointer (lambda () #f))
#t))
(define-syntax (ffi-lambda stx)
(define (make lib sym sig)
(let ((fsym (->string (->datum sym))))
#`(get-ffi-obj #,fsym #,lib #,sig (make-failure-thunk '#,sym))))
(syntax-case stx ()
((_ sym sig)
(make (->stx stx 'libcrypto) #'sym #'sig))
((_ lib sym sig)
(make #'lib #'sym #'sig))))
(define-syntax lambda/ffi
(syntax-rules (: ->)
((_ lib (f args ...))
(ffi-lambda lib f (_fun args ... -> _void)))
((_ lib (f args ...) -> type)
(ffi-lambda lib f (_fun args ... -> type)))
((_ lib (f args ...) -> type : guard)
(ffi-lambda lib f (_fun args ... -> (r : type) -> (guard 'f r))))
((_ (f args ...) rest ...)
(lambda/ffi libcrypto (f args ...) rest ...))))
(define-syntax define/ffi
(syntax-rules ()
((_ (f args ...) rest ...)
(define/ffi libcrypto (f args ...) rest ...))
((_ lib (f args ...) rest ...)
(define f (lambda/ffi lib (f args ...) rest ...)))))
(define-syntax (define/alloc stx)
(define (make lib sym)
(let ((fsym (->datum sym)))
(with-syntax
((new (->stx stx (make-symbol fsym "_new")))
(free (->stx stx (make-symbol fsym "_free"))))
#`(begin
(define new
(ffi-lambda #,lib new
(_fun -> (r : _pointer)
-> (if r r (error 'new "libcrypto: out of memory")))))
(define free
(ffi-lambda #,lib free
(_fun _pointer -> _void)))))))
(syntax-case stx()
((_ sym)
(make (->stx stx 'libcrypto) #'sym))
((_ lib sym)
(make #'lib #'sym))))
(define-syntax with-fini
(syntax-rules ()
((_ fini body ...)
(dynamic-wind
void
(lambda () body ...)
(lambda () fini)))))
(define-syntax let/fini
(syntax-rules ()
((_ () body ...) (begin body ...))
((_ ((var exp) . rest) body ...)
(let ((var exp))
(let/fini rest body ...)))
((_ ((var exp fini) . rest) body ...)
(let ((var exp))
(with-fini (fini var)
(let/fini rest body ...))))))
(define-syntax with-error-fini
(syntax-rules ()
((_ fini body ...)
(with-handlers*
(((lambda e #t)
(lambda (e) fini (raise e))))
body ...))))
(define-syntax let/error-fini
(syntax-rules ()
((_ () body ...) (begin body ...))
((_ ((var exp) . rest) body ...)
(let ((var exp))
(let/error-fini rest body ...)))
((_ ((var exp fini) . rest) body ...)
(let ((var exp))
(with-error-fini (fini var)
(let/error-fini rest body ...))))))
(define-syntax push!
(syntax-rules ()
((_ var obj) (set! var (cons-immutable obj var)))))
(define call/values call-with-values)
(let ()
(define/ffi (ERR_load_crypto_strings))
(define/ffi (OpenSSL_add_all_ciphers))
(define/ffi (OpenSSL_add_all_digests))
(ERR_load_crypto_strings)
(OpenSSL_add_all_ciphers)
(OpenSSL_add_all_digests))
)