module Netmcore:sig..end
This library allows it to develop parallel algorithms that can take advantage of multiple CPU cores. It does not use Ocaml's multi-threading feature because this would implicitly serialize all computations. Instead, independent processes are created, and the communication between processes is made as cheap and unproblematic as possible.
    Please read Netmcore_basics for an introduction into using
    this module. There is also Netmcore_tut with more advanced
    techniques.
    Before using this library, it is required to call Netmcore.startup.
    This turns the current process into the master process. The master
    process has only a supervising function, and is responsible for
    managing global resources, and for starting further worker processes.
    The worker processes can start more workers. This is different from
    creating a new thread, though: The new worker does not share memory
    with its creator. It is also different from calling Unix.fork,
    because new workers are always subprocesses of the master process.
    This means it is not initialized with a copy of the state of the
    logical creator, but with its real parent process which is always
    the master.
Compatibility with multi-threading: You may run into big trouble when the master process starts further threads (after forking thread-related resources are in an uncontrolled state). There is no such problem in the worker processes. In this library, however, nothing is done to ease the multi-threaded life, so you should carefully check the compatibility first.
    Netmcore controls the lifetime of certain system resources in
    addition to worker processes:
disable_pmanage parameter for the Netmcore.startup call, or by
      setting theunlink parameter in the same call. This second option only disables
      the deletion, but the management file is updated.typeres_id =[ `Resource of int ]
typeprocess_id =[ `Process of int ]
      Note that the int argument is not the Unix PID, but just a self-generated
      identifier that is unique for the lifetime of the program.
typecompute_resource_type =[ `File
| `Fork_point
| `Join_point
| `Posix_sem
| `Posix_shm
| `Posix_shm_preallocated
| `Posix_shm_preallocated_sc
| `Posix_shm_sc ]
typeinherit_request =[ `All | `Resources of res_id list ]
typecompute_resource_repr =[ `File of string
| `Fork_point of
inherit_request * Netplex_encap.encap -> process_id
| `Join_point of process_id -> Netplex_encap.encap option
| `Posix_sem of string
| `Posix_shm of string
| `Posix_shm_preallocated of string * Netsys_mem.memory
| `Posix_shm_preallocated_sc of
string * Netsys_mem.memory * Netsys_sem.container
| `Posix_shm_sc of string * Netsys_sem.prefix ]
`File name: (Temporary) file name (absolute name)`Posix_shm name: Shared memory objects with name`Posix_shm_sc(name,p): Shared memory objects with name, and
        attached container for semaphores with prefix p`Posix_shm_preallocated(name,m): Shared memory objects already
        allocated by the master process. These objects are passed over
        to the worker processes by inheritance, and are always mapped at
        the same address. m is the bigarray mapping the object.`Posix_shm_preallocated_sc(name,m.p): Same, plus attached
        container for semaphores with prefix p`Posix_sem name: Semaphores with name`Fork_point(inh,f): Fork points where let pid=f arg fork a new process
        with argument arg. pid is the process identifier. The list inh
        are resources inherited from the master.`Joint_point f: Joint points where let res=f pid wait until
        the process pid terminates. If res is non-None it is the
        result value. If it is None, no result was passed back
        (including all pathological cases like crashes)Netplex_cenv.pmanage and
      Netsys_pmanage).exception No_resource of res_id
class type compute_resource =object..end
Netmcore_process
    is easier to use (and less verbose).val def_process : (Netplex_encap.encap -> Netplex_encap.encap) ->
       res_id * res_idlet fork_point, join_point = def_process f:
      Defines process types, i.e. ways of starting and finishing processes. 
      The definition must be done in the master process, e.g. before the first
      worker is started. 
      Once the process type is defined, new processes can be started,
      and these processes will run the function f. When f is 
      finished, the process will terminate. Starting processes is possible
      from other worker processes.
      f is supplied with its argument by the process starter start.
      The result value of f can be retrieved with join (by any
      process, but only the first join for this process will be successful).
      If f throws exceptions, these will be caught and logged, but not
      be passed back to the caller (which just gets None as result
      of join). The same happens for any unplanned termination of the
      process.
      It is allowed to immediately release the join_point if there is
      no interest in catching the termination of started processes.
      Here is an example how to define a process that takes a string
      argument and returns an int:
      module String_encap = Netplex_encap.Make_encap(struct type t=string end)
      module Int_encap = Netplex_encap.Make_encap(struct type t=int end)
      let my_process_fork, my_process_join =
        Netmcore.def_process
          (fun s_encap ->
            let (s:string) = String_encap.unwrap s_encap in
            let (r:int) = ... in
            Int_encap.wrap r
          )
      
      The wrapping and unwrapping is required for ensuring type-safety
      (see Netplex_encap for the details of the idea).
Calling this process is done with (also see below):
      let pid = Netmcore.start my_process_fork (String_encap.wrap s) in
      let r_encap_opt = Netmcore.join my_process_join pid in
      match r_encap_opt with
        | None -> failwith "Something went wrong"
        | Some r_encap -> Int_encap.unwrap r_encap
      val start : ?inherit_resources:inherit_request ->
       res_id -> Netplex_encap.encap -> process_idlet pid = start fork_point arg: Starts the process with the
      given fork_point and the argument arg.
      Raises No_resource if there is no such resource.
The function returns a process identifier. This is not the Unix PID, but a sequentially generated number that is unique for a running program.
      Option inherit_resources: Certain resources are only accessible by
      the process when they are inherited to it. This is the case for
      `Posix_shm_preallocated. This can be set to `All to inherit
      all inheritable resources, or to `Resources l to only inherit
      the resources of l. By default, all resources are inherited.
      (This changed in Ocamlnet-3.5 - before, no resources were inherited,
      which turned out to be quite dangerous as default.)
val join : res_id -> process_id -> Netplex_encap.encap optionlet res_opt = join join_point pid: Waits until the process pid
      is done, and returns the result value if any. You can only join
      (or join_nowait) once per process.
      Raises No_resource if there is no such resource.
      It is not possible to call join from the master process (as this
      process must not block). You can, however, get the result of an
      already terminated process with join_nowait, e.g. after
      Netmcore.run returned.
val join_nowait : res_id -> process_id -> Netplex_encap.encap optionlet res_opt = join_nowait join_point pid: Checks if the process pid
      is done, and returns the result value if so. Otherwise return None.val get_resource : res_id -> compute_resourcerelease when
      your are done with the resource.
      Raises No_resource if there is no such resource.
val release : res_id -> unitrelease
      method on the object.)val manage_file : string -> compute_resourceval get_file : res_id -> stringNo_resource). As for
      get_resource the file is marked as being used by the process.val manage_shm : string -> compute_resourceval manage_shm_sc : string -> Netsys_sem.container -> compute_resourceval get_shm : res_id -> stringNo_resource). As
      for get_resource the shm object is marked as being used by the process.val get_sem_container : res_id -> Netsys_sem.containerNetsys_posix.shm_create,
    and opened with Netsys_posix.shm_open.val create_preallocated_shm : ?value_area:bool -> string -> int -> res_id * stringcreate_preallocated_shm prefix size: Creates a new preallocated
      shm object with a unique name based on prefix, and a length of
      size bytes. The object is created and mapped into the master
      process, and will be available to any newly started process when
      the resource ID is inherited to the process.
      Returns (res_id,shm_name) where res_id identifies the new
      resource, and shm_name is the name of the POSIX shared memory
      object.
      Note that the process calling this function cannot look up this
      resource (using get_shm or get_resource) because the shm
      block cannot be mapped at the right address. Nevertheless, the calling
      process counts as a user of the object, and needs to release
      the object.
      Option value_area: if set, the new memory is marked as value
      area, so the ocaml runtime allows value comparisons in this
      memory area.
val create_preallocated_shm_sc : ?value_area:bool ->
       string -> int -> res_id * string * Netsys_sem.containercreate_preallocated_shm with the extension that 
      a semaphore container is also allocated and returnedval manage_sem : string -> compute_resourceval get_sem : res_id -> stringNo_resource). As
      for get_resource the semaphore is marked as being used by the process.Netsys_posix.sem_open, and
    created with Netsys_posix.sem_create.val self_process_id : unit -> process_idval add_plugins : Netplex_types.controller -> unitpost_add_hook of the processor.val startup : socket_directory:string ->
       ?pidfile:string ->
       ?init_ctrl:(Netplex_types.controller -> unit) ->
       ?disable_pmanage:bool ->
       ?no_unlink:bool -> first_process:('a -> process_id) -> 'a -> unitstartup function returns first when this
      process is finished, in which case the whole Netplex system is
      shut down (which may lead to killing the remaining processes,
      following the usual shutdown procedure).
      The first process is created by calling first_process at the
      right moment. This function normally just invokes start.
      Passing a socket_directory is mandatory. This directory will
      contain helper files. The must be a separate socket_directory
      for each running Computeplex instance.
      pidfile: If passed, the Unix PID of the master process is written
      to this file.
      disable_pmanage: Disable that persistent kernel objects are
      managed via the "netplex.pmanage" file in the socket directory.
      no_unlink: Disable that old persistent kernel objects are deleted at
      startup. This may be useful when startup is called a second time
      from the same program.
val run : socket_directory:string ->
       ?pidfile:string ->
       ?init_ctrl:(Netplex_types.controller -> unit) ->
       ?disable_pmanage:bool ->
       ?no_unlink:bool ->
       first_process:('a -> process_id) ->
       extract_result:(Netplex_types.controller -> process_id -> 'b) ->
       'a -> 'bstartup, namely that it was
      difficult to get the result of the process(es) back. Here, the
      additional callback extract_result can be used to retrieve the
      results from processes, especially from the first process.
      You need to call Netmcore.join_nowait to get the results from
      processes.
      The result of extract_result is the result of run.
val destroy_resources : unit -> unitmodule Debug:sig..end