FFI-UCtags for Ruby is a utility gem that loads an FFI library by reading a list of to-import constructs off a C header file.

This gem is still developing; but once mature, Ruby ports for C libraries are free from the duty of tracking the APIs manually! Just download the header, find a prebuilt shared library (or two if going multi-platform), and feed them into this utility. Maybe complement with a few Ruby scripts to enhance the OOP convenience, and boom, libXXX ported in less than an hour!

Caution: Currently, this project does not have automated testing (also insights welcome!), instead relies on code review and small test subjects.

Example: QOI

phoboslab/qoi@dfc056e

require 'ffi-uctags'

# Import QOI library in one line
QOI = FFI::UCtags.call 'path/to/libqoi.so', 'path/to/qoi.h'

# Build a Struct
meta = QOI::Qoi_desc.new
meta[:width] = meta[:height] = 3
meta[:channels] = 4
meta[:colorspace] = 0 # QOI_SRGB

# Use the library like how you would with manually-imported FFI
bytes = 0
FFI::MemoryPointer.new(:uint32, 9) do|pixels|
  pixels.write_array_of_uint32 [
    # AABBGGRR (most architectures store integers in little-endian)
    0x00000000, 0xFF000000, 0xFFFFFFFF,
    0xFF0000FF, 0xFF00FF00, 0xFFFF0000,
    0xFF00FFFF, 0xFFFF00FF, 0xFFFFFF00
  ]
  bytes = QOI.qoi_write('path/to/output.qoi', pixels, meta)
end

puts "Written #{bytes} bytes"
exit !bytes.zero? # `#qoi_write` returns 0 on failure

Setup

Dependency: Universal Ctags

Universal Ctags (u-ctags) does all the heavy-lifting of parsing the header; this gem merely parses and processes its output. (Therefore, most of the credits goes to the u-ctags team!)

For your convenience, this gem bundles the u-ctags repository as a submodule at /u-ctags/src/ in the gem, and rake default (which gem install invokes) builds the executable at /u-ctags/bin.

To whom it may concern: A reminder that, the submodule is licensed under GPL-2.0, separate from this repository’s Apache-2.0. This licensing with its “include the source code” requirement is why I decided against just bundling their prebuilt ctags executable.

Install

[Command TBA]

As stated above, this builds the bundled (cough submoduled cough) u-ctags right inside the gem directory.

For Development

rake setup

This is the same as:

rake default bundle

That is, after building u-ctags, follow up with the bundle task, which gets rid of the u-ctags submodule (via git submodule deinit if applicable) and then bundle install. This is because otherwise Bundler would wanna process the submodule’s zillions of source files, making every invocation painfully slow. If for some integration development purposes that you’d like to retain them, simply rake the first default task, smuggle the sources out of the Git repository, then resume with the rake bundle task (or just your usual bundle install workflow if you don’t need the task’s cleanup.)

Distributing – packaging RubyGem or vendoring (bundling)

Make sure to include the u-ctags source code at where it was when you downloaded this repo. Thing is, even if we include a prebuilt so your clients don’t need the sources to build themselves, remember that we still gotta comply with that source code requirement of u-ctags’s GPL-2.0 (GGWP, GPL).

If you checked the repository out via Git/Hub, simply reload the submodule with the following or similar command. If not – you’ll have to re-download if you don’t have a backup (good luck).

git submodule update u-ctags

Features

Constructs & Ctags kinds support

☑️️ Developed

  • Recognition of basic C types (unsigned char, int8_t, etc.)
  • Function Prototypes
    • p function prototypes
    • f function definitions
    • z function parameters inside function or prototype definitions
  • Enums
    • e enumerators (values inside an enumeration)
    • g enumeration names
  • Miscellaneous
    • t typedefs
    • x external and forward variable declarations

📝 Developing

  • Structs/Unions
    • m struct, and union members
    • s structure names
    • u union names
    • nested structs and unions
    • currently does not understand array members (#14)
    • currently does not recognize opaque structs/unions (#16)

🔜 To Do

  • Literal Macros (macro-defined constants) (#2)
    • d macro definitions
  • FFI callbacks (wraps pointer to functions) (#3)
  • Variadic args (#4)
  • Import referenced headers (i.e., nested imports) (#5)
    • h included header files

⏳ No Plans Yet

  • FFI Types :string, :strptr and :buffer_*
  • Enums that aren’t simply 0...size
    • Let me or the u-ctags team know if this is a much-wanted feature.
  • Structs/unions defined inside functions’ parameter list
    • E.g., void dubious_function(struct { … } data);
    • They are not recognized by u-ctags
    • “C allows struct, union, and enum types to be declared in function prototypes, whereas C++ does not.” ⸺ Wikipedia
  • Parameterized Macros
    • D parameters inside macro definitions

🧊 Nope

  • Non-literal Macros (i.e., C code macros)
  • Miscellaneous u-ctags Kinds
    • v variable definitions
      • Unlike x, these are not exported to dynamic libraries (.sos).
    • l local variables
    • L goto labels

Additional capabilities

  • Passive design enables working with alternate FFI implementations such as Nice-FFI

Regarding structs and unions

Structs and unions are classes in FFI, thus this gem chooses to import them as constants. Top-level structs/unions are under the imported FFI::Library’s namespace, while inner structs/unions nest under outer ones.

Whereas FFI imports typedefs as Symbol keys of a table of types, this gem handles typedefs of struct/unions specially to import them as constants. The gem prefers typedef aliases over original names, which is often omitted though the typedef-struct and typedef-union patterns. For example:

struct MyStruct { // Named, but must prefix `struct ` every use (`struct MyStruct`)
  …
}
typedef struct { // No name
  …
} MyStruct_t; // The name `MyStruct_t` actually belongs to a typedef.

This gem imports the first struct as MyStruct and the second as MyStruct_t (rather than some anonymous id generated by u-ctags).

Since Ruby constants must start with an uppercase letter, this gem capitalizes the first char for names that don’t meet the criterion, or prefix with S or U for ones that don’t start with a capitalizable char (typically _). For example, the struct in the example above is typedef-named qoi_desc in the original header; it becomes Qoi_desc to meet the capitalization criterion.

Structs and unions with neither a name nor typedef aliases use placeholder names generated by u-ctags, which is __anon### where ### is a hash ID, thus you’d find them around with S__anon or U__anon prefixes. The hash is consistent as long as the path/to/header.h is the same. See: https://github.com/universal-ctags/ctags/blob/v6.0.0/docs/parser-cxx.rst#anonymous-structure-names

U-ctags limitation: Macros

U-ctags is not a C preprocessor. It currently only follows preprocessing directives naïvely. Preprocessor macros can confuse u-ctags (and consequently this gem) to parse inappropriate constructs, especially templates that generate content.

See: universal-ctags/ctags#2356

Meanwhile, patching headers and/or preprocessing them (e.g., gcc -E) works this problem around.

License

This repository (excluding submodule(s))

Copyright 2023 ParadoxV5

Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with the License. You may obtain a copy of the License at

http://www.apache.org/licenses/LICENSE-2.0

Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the specific language governing permissions and limitations under the License.

Universal Ctags submodule (/u-ctags/src/)

Universal Ctags, Copyright (C) 2015-2022 Universal Ctags Team Universal Ctags is derived from Exuberant Ctags. Exuberant Ctags 5.8, Copyright (C) 1996-2009 Darren Hiebert URL: https://ctags.io/

This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 2of the License, or (at your option) any later version.

This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details.

You should have received a copy of the GNU General Public License along with this program; if not, write to the Free Software Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.