shared-object.scm 3.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596
  1. ; Copyright (c) 1993-2008 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. (define (shared-object-address-or-false shared-object name)
  62. (guard (c (else #f))
  63. (shared-object-address shared-object name)))
  64. ;; This simply calls a C function with no parameters and no return
  65. ;; value. It's typically for calling the initialization function; we
  66. ;; can't use any of the regular external-calling mechanisms because
  67. ;; they expect a s48_value return value, where the initialization
  68. ;; function has void.
  69. (define (call-shared-object-address s-o-address)
  70. (external-call-thunk (shared-object-address-value s-o-address)))
  71. (import-lambda-definition external-dlopen (name generate-name?)
  72. "shared_object_dlopen")
  73. (import-lambda-definition external-dlsym (handle name)
  74. "shared_object_dlsym")
  75. (import-lambda-definition external-dlclose (shared-object)
  76. "shared_object_dlclose")
  77. (import-lambda-definition external-call-thunk (address)
  78. "shared_object_call_thunk")