block-sys 0.2.1

Raw bindings to Apple's C language extension of blocks
Documentation
# `block-sys`

[![Latest version](https://badgen.net/crates/v/block-sys)](https://crates.io/crates/block-sys)
[![License](https://badgen.net/badge/license/MIT/blue)](https://github.com/madsmtm/objc2/blob/master/LICENSE.txt)
[![Documentation](https://docs.rs/block-sys/badge.svg)](https://docs.rs/block-sys/)
[![CI](https://github.com/madsmtm/objc2/actions/workflows/ci.yml/badge.svg)](https://github.com/madsmtm/objc2/actions/workflows/ci.yml)

**This crate is deprecated. Use `block2::ffi` instead.**

Raw Rust bindings to Apple's C language extension of blocks.

This crate is part of the [`objc2` project](https://github.com/madsmtm/objc2),
see that for related crates.


## Runtime Support

This library is a raw interface to the aptly specified [Blocks ABI][abi].
However, different runtime implementations exist and act in slightly different
ways (and have several different helper functions), the most important aspect
being that the libraries are named differently, so the linking must take that
into account.

You can choose the desired runtime by using the relevant cargo feature flags,
see the following sections (you might have to disable the default `apple`
feature first). Note that if the `objc-sys` crate is present in the module
tree, this should have the same feature flag enabled as that.


[abi]: https://clang.llvm.org/docs/Block-ABI-Apple.html


### Apple's [`libclosure`]https://github.com/apple-oss-distributions/libclosure

- Feature flag: `apple`.

This is the most sophisticated runtime, and it has quite a lot more features
than the specification mandates. It is used by default.

The minimum required operating system versions are as follows:
- macOS: `10.6`
- iOS: `3.2`
- tvOS: Unknown
- watchOS: Unknown

Though in practice Rust itself requires higher versions than this.


### LLVM `compiler-rt`'s [`libBlocksRuntime`]https://github.com/llvm/llvm-project/tree/release/13.x/compiler-rt/lib/BlocksRuntime

- Feature flag: `compiler-rt`.

This is a copy of Apple's older (around macOS 10.6) runtime, and is now used
in [Swift's `libdispatch`] and [Swift's Foundation] as well.

The runtime and associated headers can be installed on many Linux systems with
the `libblocksruntime-dev` package.

Using this runtime probably won't work together with `objc-sys` crate.

[Swift's `libdispatch`]: https://github.com/apple/swift-corelibs-libdispatch/tree/swift-5.5.1-RELEASE/src/BlocksRuntime
[Swift's Foundation]: https://github.com/apple/swift-corelibs-foundation/tree/swift-5.5.1-RELEASE/Sources/BlocksRuntime


### GNUStep's [`libobjc2`]https://github.com/gnustep/libobjc2

- Feature flag: `gnustep-1-7`, `gnustep-1-8`, `gnustep-1-9`, `gnustep-2-0` and
  `gnustep-2-1` depending on the version you're using.

GNUStep is a bit odd, because it bundles blocks support into its Objective-C
runtime. This means we have to link to `libobjc`, and this is done by
depending on the `objc-sys` crate. A bit unorthodox, yes, but it works.

Sources:
- [`Block.h`]https://github.com/gnustep/libobjc2/blob/v2.1/objc/blocks_runtime.h
- [`Block_private.h`]https://github.com/gnustep/libobjc2/blob/v2.1/objc/blocks_private.h


### Microsoft's [`WinObjC`]https://github.com/microsoft/WinObjC

- Feature flag: `unstable-winobjc`.

**Unstable: Hasn't been tested on Windows yet!**

[A fork](https://github.com/microsoft/libobjc2) based on GNUStep's `libobjc2`
version 1.8.


### [`ObjFW`]https://github.com/ObjFW/ObjFW

- Feature flag: `unstable-objfw`.

**Unstable: Doesn't work yet!**

TODO.


## C Compiler configuration

To our knowledge, currently only `clang` supports the [Language Specification
for Blocks][block-lang]. To assist in compiling C (or Objective-C) sources
using these features, this crate's build script expose the
`DEP_BLOCK_0_2_CC_ARGS` environment variable to downstream build scripts.

Example usage in your `build.rs` (using the `cc` crate) would be as follows:

```rust , ignore
fn main() {
    let mut builder = cc::Build::new();
    builder.compiler("clang");
    builder.file("my_script_using_blocks.c");

    for flag in std::env::var("DEP_BLOCK_0_2_CC_ARGS").unwrap().split(' ') {
        builder.flag(flag);
    }

    builder.compile("libmy_script_using_blocks.a");
}
```

[block-lang]: https://clang.llvm.org/docs/BlockLanguageSpec.html