package ctypes

  1. Overview
  2. Docs
Combinators for binding to C libraries without writing any C

Install

dune-project
 Dependency

Authors

Maintainers

Sources

0.24.0.tar.gz
sha256=249c5604c8554659761a7282db4ad200aa8c0fdc408cdb53102bd70feeb51955
md5=064316aaf508a9db203f114bb8401dee

Description

ctypes is a library for binding to C libraries using pure OCaml. The primary aim is to make writing C extensions as straightforward as possible. The core of ctypes is a set of combinators for describing the structure of C types -- numeric types, arrays, pointers, structs, unions and functions. You can use these combinators to describe the types of the functions that you want to call, then bind directly to those functions -- all without writing or generating any C!

To install the optional ctypes-foreign interface (which uses libffi to provide dynamic access to foreign libraries), you will need to also install the ctypes-foreign package.

opam install ctypes-foreign

This will make the ctypes-foreign ocamlfind subpackage available.

Tags

org:mirage

Published: 13 Nov 2025

README

ctypes is a library for binding to C libraries using pure OCaml. The primary aim is to make writing C extensions as straightforward as possible.

The core of ctypes is a set of combinators for describing the structure of C types -- numeric types, arrays, pointers, structs, unions and functions. You can use these combinators to describe the types of the functions that you want to call, then bind directly to those functions -- all without writing or generating any C!

GitHub Actions status

Usage

Suppose you want to bind to the following C functions:

   int sigemptyset(sigset_t *set);
   int sigfillset(sigset_t *set);
   int sigaddset(sigset_t *set, int signum);
   int sigdelset(sigset_t *set, int signum);
   int sigismember(const sigset_t *set, int signum);

Using ctypes you can describe the interfaces to these functions as follows:

   let sigemptyset = foreign "sigemptyset" (ptr sigset_t @-> returning int)
   let sigfillset = foreign "sigfillset" (ptr sigset_t @-> returning int)
   let sigaddset = foreign "sigaddset" (ptr sigset_t @-> int @-> returning int)
   let sigdelset = foreign "sigdelset" (ptr sigset_t @-> int @-> returning int)
   let sigismember = foreign "sigismember" (ptr sigset_t @-> int @-> returning int)

The names bound by this code have the types you might expect:

   val sigemptyset : sigset_t ptr -> int
   val sigfillset : sigset_t ptr -> int
   val sigaddset : sigset_t ptr -> int -> int
   val sigdelset : sigset_t ptr -> int -> int
   val sigismember : sigset_t ptr -> int -> int

That's all there is to it. Unlike the usual way of writing C extensions, there are no C "stub" functions to write, so there's much less opportunity for error.

The documentation and source distribution contain more complex examples, involving structs, unions, arrays, callback functions, and so on, and show how to create and use C values (like instances of sigset_t ptr) in OCaml.

Dependencies (4)

  1. dune-configurator
  2. integers
  3. ocaml >= "4.07.0"
  4. dune >= "3.9"

Dev Dependencies (4)

  1. odoc with-doc
  2. conf-pkg-config with-test
  3. conf-fts with-test & os != "win32"
  4. ounit2 with-test

  1. ahrocksdb
  2. antic < "0.4.0"
  3. arb < "0.4.0"
  4. argon2
  5. arrayjit
  6. async_ssl >= "v0.15.0"
  7. bimage-unix
  8. bitgenerators
  9. calcium < "0.4.0"
  10. camlkit-base
  11. ceph
  12. cf
  13. checked_oint < "0.4.1"
  14. cmark
  15. cmarker
  16. corosync
  17. ctypes-build
  18. ctypes-foreign >= "0.24.0"
  19. ctypes-zarith
  20. ctypes_stubs_js
  21. cudajit
  22. decompress >= "1.3.0" & < "1.5.3"
  23. directories
  24. dlm >= "0.3.3"
  25. dune >= "3.17.2"
  26. echo
  27. eigen >= "0.1.4"
  28. extism
  29. flint
  30. fsevents
  31. gccjit
  32. gdal >= "0.10.0"
  33. gl-legacy
  34. gobject-introspection
  35. gpiod
  36. gr
  37. guile
  38. hacl-star-raw >= "0.7.2"
  39. hardcaml_c
  40. hardcaml_verilator
  41. hdr_histogram
  42. imguiml
  43. kqueue < "0.2.0"
  44. libbpf
  45. libbpf_maps
  46. libdash >= "0.3"
  47. libdrm
  48. libirmin
  49. lilv
  50. llama-cpp-ocaml
  51. llvm >= "3.7"
  52. lp-glpk
  53. lp-gurobi
  54. luv
  55. luv_unix
  56. lz4 >= "1.3.0"
  57. mariadb < "1.1.5" | >= "1.2.0"
  58. memcpy >= "0.2.2"
  59. metal
  60. mmdb
  61. mpg123
  62. netlink >= "0.3.4"
  63. nx < "1.0.0~alpha1"
  64. ocaml-yescrypt
  65. octez-l2-libs
  66. octez-libs
  67. opasswd
  68. opencc < "transition"
  69. opencc0
  70. opencc1
  71. opencc1_1
  72. orocksdb
  73. osx-acl
  74. osx-attr
  75. osx-cf
  76. osx-fsevents
  77. osx-membership
  78. osx-mount
  79. osx-secure-transport
  80. osx-xattr
  81. owl >= "0.9.0"
  82. pari
  83. pari-bindings
  84. pg_query
  85. pkcs11-driver
  86. pkcs11-rev
  87. portaudio_c_bindings
  88. portmidi
  89. posix-base
  90. posix-bindings < "3"
  91. posix-errno
  92. posix-getopt >= "2.0.0"
  93. posix-math2
  94. posix-resource
  95. posix-signal
  96. posix-socket
  97. posix-socket-unix
  98. posix-stat
  99. posix-time2
  100. posix-types
  101. posix-uname
  102. posix-unistd
  103. ppx_cstubs >= "0.7.0"
  104. py >= "1.1"
  105. quickjs
  106. raygui < "1.5.1"
  107. raylib
  108. reed-solomon-erasure
  109. rfc1951 >= "1.3.0"
  110. rune
  111. sanlock
  112. sarek
  113. sattools
  114. sodium < "0.6.0"
  115. spin >= "0.8.0"
  116. spoc
  117. srt
  118. stk_iconv
  119. swipl
  120. sys-socket
  121. sys-socket-unix
  122. tensorflow
  123. tezos-hacl
  124. tezos-hacl-glue-unix < "12.3"
  125. tezos-sapling
  126. tezos-wasmer
  127. tgls >= "0.9.0"
  128. tinyfiledialogs
  129. torch
  130. tsdl >= "1.1.0"
  131. tsdl-image >= "0.3.0"
  132. tsdl-mixer
  133. tsdl-ttf >= "0.3"
  134. unix-errno >= "0.4.1" & < "0.5.0" | >= "0.6.2"
  135. unix-sys-resource
  136. unix-sys-stat
  137. unix-time
  138. unix-type-representations >= "0.1.1"
  139. uring-trace
  140. wasmer
  141. wasmtime
  142. xedbindings
  143. xxhash
  144. yaml
  145. yara
  146. yices2_bindings
  147. zstandard
  148. zstd >= "0.4"

Conflicts (1)

  1. host-system-msvc