shared-object.scm 3.2 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192
  1. ; Copyright (c) 1993-2007 by Richard Kelsey and Jonathan Rees. See file COPYING.
  2. ; This is file shared-object.scm.
  3. ; Dynamically load external object files.
  4. (define-record-type shared-object :shared-object
  5. (make-shared-object name complete-name? c-handle)
  6. shared-object?
  7. (name shared-object-name)
  8. ;; says whether the OS should add a system-dependent extension
  9. ;; (such as .so or .dll) or do some other such transformation
  10. (complete-name? shared-object-complete-name?)
  11. (c-handle shared-object-c-handle
  12. set-shared-object-c-handle!))
  13. (define-record-discloser :shared-object
  14. (lambda (shared-object)
  15. (list 'shared-object
  16. (shared-object-name shared-object))))
  17. ;; Doing better would be quite a bit of work: Each shared object would
  18. ;; need to maintain a population of addresses derived from it, so we
  19. ;; can avoid ordering problems. Morever, the filename specified might
  20. ;; be relative, causing further problems.
  21. ;; On the other hand, the high-level code might have a better approach
  22. ;; to this, so we don't forbid dumping on these.
  23. (define-record-resumer :shared-object #t)
  24. (define (open-shared-object name complete-name?)
  25. (let ((shared-object (make-shared-object name
  26. complete-name?
  27. (external-dlopen (os-string->byte-vector
  28. (x->os-string name))
  29. complete-name?))))
  30. (add-finalizer! shared-object close-shared-object)
  31. shared-object))
  32. (define (close-shared-object shared-object)
  33. (let ((c-handle (shared-object-c-handle shared-object)))
  34. (if c-handle
  35. (begin
  36. (external-dlclose c-handle)
  37. (set-shared-object-c-handle! shared-object #f)))))
  38. (define-record-type shared-object-address :shared-object-address
  39. (make-shared-object-address object
  40. name
  41. value)
  42. shared-object-address?
  43. (object shared-object-address-object)
  44. (name shared-object-address-name)
  45. (value shared-object-address-value))
  46. (define-record-discloser :shared-object-address
  47. (lambda (shared-object-address)
  48. (list 'shared-object-address
  49. (shared-object-address-object shared-object-address)
  50. (shared-object-address-name shared-object-address))))
  51. (define-record-resumer :shared-object-address #f)
  52. (define (shared-object-address shared-object name)
  53. (make-shared-object-address shared-object
  54. name
  55. (external-dlsym (shared-object-c-handle shared-object)
  56. (os-string->byte-vector
  57. (call-with-os-string-text-codec
  58. utf-8-codec
  59. (lambda ()
  60. (x->os-string name)))))))
  61. ;; This simply calls a C function with no parameters and no return
  62. ;; value. It's typically for calling the initialization function; we
  63. ;; can't use any of the regular external-calling mechanisms because
  64. ;; they expect a s48_value return value, where the initialization
  65. ;; function has void.
  66. (define (call-shared-object-address s-o-address)
  67. (external-call-thunk (shared-object-address-value s-o-address)))
  68. (import-lambda-definition external-dlopen (name generate-name?)
  69. "shared_object_dlopen")
  70. (import-lambda-definition external-dlsym (handle name)
  71. "shared_object_dlsym")
  72. (import-lambda-definition external-dlclose (shared-object)
  73. "shared_object_dlclose")
  74. (import-lambda-definition external-call-thunk (address)
  75. "shared_object_call_thunk")