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
require 'ffi-uctags'
# Import QOI library in one line
QOI = FFI::UCtags.call 'path/to/libqoi.so', 'path/to/qoi.h'
# Build a Struct
= QOI::Qoi_desc.new
[:width] = [:height] = 3
[:channels] = 4
[: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, )
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 prototypesf
function definitionsz
function parameters inside function or prototype definitions
- Enums
e
enumerators (values inside an enumeration)g
enumeration names
- Miscellaneous
t
typedefsx
external and forward variable declarations
📝 Developing
- Structs/Unions
🔜 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
, andenum
types to be declared in function prototypes, whereas C++ does not.” ⸺ Wikipedia
- E.g.,
- 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 (.so
s).
- Unlike
l
local variablesL
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
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.