rcx 0.3.0 → 0.4.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
checksums.yaml CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: 67bf53ccbe670ee75a0faa064fed2c9cedb0f91042c52a61f75d2fa2948f36b7
4
- data.tar.gz: cc83c45e03bd7cb72bf8856385c74c061e3995c50f409ff9fbc9beb9a62c12c2
3
+ metadata.gz: 422a1c5bd974518c00e3a6c0ca86ac38f6c62ff06dd56acf0971973f215208bc
4
+ data.tar.gz: 8ecbbb1e288c7df1ebdb317f64c1c2656faf50044c4e6d06c598a6789d21906d
5
5
  SHA512:
6
- metadata.gz: '08a0a98e5eade9330c4cd599b25264a3ea792ff5fc7067a122176bbf0a95e766cb6b7d0e6bc583e1bdb8535d49f5ac36624453bfa6455b381564b413e73ee5bb'
7
- data.tar.gz: 19d7231302d29d691af29fd21696e43ed9f31be3b293f48c87fc7cd6b73e58e23307a558693cdb60da49d7d8a2a6c553c35bdbb6def1e0efd4ccf847cc07e2da
6
+ metadata.gz: ec21a72b07c85b513459908b61733ee68c61c3116729ed87209a603666ed2396edb0ab140449db0d16185a183215dfb750c5e9ae8f5af5c3652accd63196dad6
7
+ data.tar.gz: 4d3bbb65011e6e2649da175130f22cd9bcf4bbbd95830452e06bad119d02b3904e2e514bc4aac05f92e32729bc463faa865ea6d6c4bccbd061a00b39cedb8d7a
data/CHANGELOG.md CHANGED
@@ -1,5 +1,11 @@
1
1
  ## UNRELEASED
2
2
 
3
+ ## v0.4.0 (2025-09-03)
4
+ - Renamed `rcx::arg` namespace to `rcx::args` to prevent conflit with `rcx::arg::arg`.
5
+
6
+ ## v0.3.1 (2025-09-02)
7
+ - Now `extconf.rb`'s can only `require 'rcx/mkmf/c++20'` to enable RCX.
8
+
3
9
  ## v0.3.0 (2025-08-22)
4
10
  - Added `rcx::gvl::without_gvl`
5
11
 
data/README.md CHANGED
@@ -6,9 +6,7 @@ Write Ruby extensions in C++20. Inspired by [rice](https://github.com/ruby-rice/
6
6
  ## Creating a new extension
7
7
  In your `extconf.rb` file, add the following:
8
8
  ```ruby
9
- require 'mkmf
10
- $CXXFLAGS += ' -std=c++20' # or newer
11
- require 'rcx/mkmf'
9
+ require 'rcx/mkmf/c++20'
12
10
 
13
11
  create_header
14
12
  create_makefile('your_ext')
data/Rakefile CHANGED
@@ -39,6 +39,6 @@ task :yarn_install do
39
39
  end
40
40
 
41
41
  desc 'Format the source code'
42
- task :format => FileList['**/*.hpp'] do |t|
42
+ task :format => FileList['**/*.[ch]pp'] do |t|
43
43
  sh 'clang-format', '-i', *t.prerequisites
44
44
  end
data/examples/class.cpp CHANGED
@@ -3,7 +3,7 @@
3
3
  #include <rcx/rcx.hpp>
4
4
 
5
5
  void Init_examples_class() {
6
- using namespace rcx::arg;
6
+ using namespace rcx::args;
7
7
 
8
8
  auto ruby = rcx::Ruby::get();
9
9
  auto cls =
@@ -67,6 +67,10 @@ namespace rcx {
67
67
  /// Concepts
68
68
  ///
69
69
  namespace concepts {
70
+ /// Specifies the type `void`.
71
+ template <typename T>
72
+ concept Void = std::is_void_v<T>;
73
+
70
74
  template <typename T>
71
75
  concept StringLike = requires {
72
76
  typename std::remove_cvref_t<T>::value_type;
@@ -286,7 +290,7 @@ namespace rcx {
286
290
  /// Specifies the types that can be converted from Ruby values.
287
291
  ///
288
292
  template <typename T>
289
- concept ConvertibleFromValue = requires(Value v) { from_Value<T>(v); };
293
+ concept ConvertibleFromValue = !std::is_void_v<T> && requires(Value v) { from_Value<T>(v); };
290
294
 
291
295
  /// Specifies the types that can be converted into Ruby values.
292
296
  ///
@@ -316,7 +320,7 @@ namespace rcx {
316
320
 
317
321
  /// Argument parsing.
318
322
  ///
319
- namespace arg {
323
+ namespace args {
320
324
  template <concepts::ConvertibleFromValue T = Value> struct Self {
321
325
  using ResultType = detail::wrap_ref_t<T>;
322
326
  static ResultType parse(Ruby &, Value self, std::span<Value> &args);
@@ -534,10 +538,40 @@ namespace rcx {
534
538
  ClassT<Derived> get_class() const;
535
539
  Derived freeze() const;
536
540
 
541
+ /// Defines a singleton method for the object.
542
+ ///
543
+ /// @warning Defining a method this way allocates a resource that will never be
544
+ /// garbage-collected.
545
+ ///
546
+ /// @tparam Self The type of the receiver.
547
+ /// @tparam ArgSpec The types of the method arguments.
548
+ /// @param mid The name of the method.
549
+ /// @param function The C++ function to be invoked. It should accept `Self` as
550
+ /// the first argument, followed by the arguments defined by `argspec`.
551
+ /// @param argspec The argument specifications for the method.
552
+ /// @return Returns the receiver.
537
553
  template <concepts::ConvertibleFromValue Self = Derived, concepts::ArgSpec... ArgSpec>
538
554
  Derived define_singleton_method(concepts::Identifier auto &&mid,
539
555
  std::invocable<Self, typename ArgSpec::ResultType...> auto &&function,
540
556
  ArgSpec... argspec) const;
557
+
558
+ /// @overload
559
+ ///
560
+ /// This overload defines a method that does not use the receiver.
561
+ ///
562
+ /// @warning Defining a method this way allocates a resource that will never be
563
+ /// garbage-collected.
564
+ ///
565
+ /// @tparam Self Must be `void`.
566
+ /// @param mid The name of the method.
567
+ /// @param function The C++ function to be invoked. It should accept the
568
+ /// arguments defined by `argspec`.
569
+ /// @param argspec The argument specifications for the method.
570
+ /// @return Returns the receiver.
571
+ template <concepts::Void Self, concepts::ArgSpec... ArgSpec>
572
+ Derived define_singleton_method(concepts::Identifier auto &&mid,
573
+ std::invocable<typename ArgSpec::ResultType...> auto &&function,
574
+ ArgSpec... argspec) const;
541
575
  };
542
576
 
543
577
  class Value: public ValueT<Value, ValueBase, Nilable> {
@@ -635,6 +669,23 @@ namespace rcx {
635
669
  std::invocable<Self, typename ArgSpec::ResultType...> auto &&function,
636
670
  ArgSpec... argspec) const;
637
671
 
672
+ /// @overload
673
+ ///
674
+ /// This overload defines a method that does not use the receiver.
675
+ ///
676
+ /// @warning Defining method this way allocates a resource that will never be
677
+ /// garbage-collected.
678
+ ///
679
+ /// @tparam Self Must be `void`.
680
+ /// @param mid The name of the method.
681
+ /// @param function The function to be called.
682
+ /// @param argspec List of argument specifications.
683
+ /// @return Self.
684
+ template <concepts::Void Self, concepts::ArgSpec... ArgSpec>
685
+ Module define_method(concepts::Identifier auto &&mid,
686
+ std::invocable<typename ArgSpec::ResultType...> auto &&function,
687
+ ArgSpec... argspec) const;
688
+
638
689
  /// Checks if a constant is defined under this module.
639
690
  ///
640
691
  /// @param name Name of the constant.
@@ -235,7 +235,7 @@ namespace rcx {
235
235
  std::conditional_t<std::derived_from<T, typed_data::WrappedStructBase>, T const &, T const>;
236
236
  };
237
237
 
238
- namespace arg {
238
+ namespace args {
239
239
  template <concepts::ConvertibleFromValue T>
240
240
  inline typename Self<T>::ResultType Self<T>::parse(Ruby &, Value self, std::span<Value> &) {
241
241
  return from_Value<T>(self);
@@ -264,7 +264,7 @@ namespace rcx {
264
264
  return arg;
265
265
  }
266
266
 
267
- inline ArgSplat::ResultType ArgSplat::parse(Ruby &, Value self, std::span<Value> &args) {
267
+ inline ArgSplat::ResultType ArgSplat::parse(Ruby &, Value, std::span<Value> &args) {
268
268
  auto result = Array::new_from(args);
269
269
  args = {};
270
270
  return result;
@@ -611,7 +611,7 @@ namespace rcx {
611
611
  inline Derived ValueT<Derived, Super, nilable>::define_singleton_method(
612
612
  concepts::Identifier auto &&mid,
613
613
  std::invocable<Self, typename ArgSpec::ResultType...> auto &&function, ArgSpec...) const {
614
- auto const callback = detail::method_callback<arg::Self<Self>, ArgSpec...>::alloc(
614
+ auto const callback = detail::method_callback<args::Self<Self>, ArgSpec...>::alloc(
615
615
  std::forward<decltype(function)>(function));
616
616
  detail::protect([&]() noexcept {
617
617
  auto const singleton = ::rb_singleton_class(this->as_VALUE());
@@ -621,6 +621,21 @@ namespace rcx {
621
621
  return *static_cast<Derived const *>(this);
622
622
  }
623
623
 
624
+ template <typename Derived, std::derived_from<ValueBase> Super, Nilability nilable>
625
+ template <concepts::Void, concepts::ArgSpec... ArgSpec>
626
+ inline Derived ValueT<Derived, Super, nilable>::define_singleton_method(
627
+ concepts::Identifier auto &&mid,
628
+ std::invocable<typename ArgSpec::ResultType...> auto &&function, ArgSpec...) const {
629
+ auto const callback =
630
+ detail::method_callback<ArgSpec...>::alloc(std::forward<decltype(function)>(function));
631
+ detail::protect([&]() noexcept {
632
+ auto const singleton = ::rb_singleton_class(this->as_VALUE());
633
+ rb_define_method_id(
634
+ singleton, detail::into_ID(std::forward<decltype(mid)>(mid)), callback, -1);
635
+ });
636
+ return *static_cast<Derived const *>(this);
637
+ }
638
+
624
639
  /// Value
625
640
 
626
641
  template <concepts::ConvertibleFromValue R>
@@ -707,7 +722,18 @@ namespace rcx {
707
722
  template <concepts::ConvertibleFromValue Self, concepts::ArgSpec... ArgSpec>
708
723
  inline Module Module::define_method(concepts::Identifier auto &&mid,
709
724
  std::invocable<Self, typename ArgSpec::ResultType...> auto &&function, ArgSpec...) const {
710
- auto const callback = detail::method_callback<arg::Self<Self>, ArgSpec...>::alloc(function);
725
+ auto const callback = detail::method_callback<args::Self<Self>, ArgSpec...>::alloc(function);
726
+ detail::protect([&]() noexcept {
727
+ rb_define_method_id(
728
+ as_VALUE(), detail::into_ID(std::forward<decltype(mid)>(mid)), callback, -1);
729
+ });
730
+ return *this;
731
+ }
732
+
733
+ template <concepts::Void, concepts::ArgSpec... ArgSpec>
734
+ inline Module Module::define_method(concepts::Identifier auto &&mid,
735
+ std::invocable<typename ArgSpec::ResultType...> auto &&function, ArgSpec...) const {
736
+ auto const callback = detail::method_callback<ArgSpec...>::alloc(function);
711
737
  detail::protect([&]() noexcept {
712
738
  rb_define_method_id(
713
739
  as_VALUE(), detail::into_ID(std::forward<decltype(mid)>(mid)), callback, -1);
@@ -792,7 +818,7 @@ namespace rcx {
792
818
  inline ClassT<T> ClassT<T>::define_method(concepts::Identifier auto &&mid,
793
819
  std::invocable<T &, typename ArgSpec::ResultType...> auto &&function, ArgSpec...) const {
794
820
  auto const callback =
795
- detail::method_callback<arg::Self<detail::self_type<T>>, ArgSpec...>::alloc(
821
+ detail::method_callback<args::Self<detail::self_type<T>>, ArgSpec...>::alloc(
796
822
  std::forward<decltype(function)>(function));
797
823
  detail::protect([&]() noexcept {
798
824
  rb_define_method_id(
@@ -807,7 +833,7 @@ namespace rcx {
807
833
  std::invocable<T const &, typename ArgSpec::ResultType...> auto &&function,
808
834
  ArgSpec...) const {
809
835
  auto const callback =
810
- detail::method_callback<arg::Self<detail::self_type_const<T>>, ArgSpec...>::alloc(
836
+ detail::method_callback<args::Self<detail::self_type_const<T>>, ArgSpec...>::alloc(
811
837
  function);
812
838
  detail::protect([&]() noexcept {
813
839
  rb_define_method_id(
@@ -820,7 +846,7 @@ namespace rcx {
820
846
  template <concepts::ArgSpec... ArgSpec>
821
847
  requires std::constructible_from<T, typename ArgSpec::ResultType...>
822
848
  inline ClassT<T> ClassT<T>::define_constructor(ArgSpec...) const {
823
- auto const callback = detail::method_callback<arg::Self<Value>, ArgSpec...>::alloc(
849
+ auto const callback = detail::method_callback<args::Self<Value>, ArgSpec...>::alloc(
824
850
  typed_data::DataType<T>::template initialize<typename ArgSpec::ResultType...>);
825
851
  detail::protect([&]() noexcept {
826
852
  using namespace literals;
@@ -833,7 +859,7 @@ namespace rcx {
833
859
  inline ClassT<T> ClassT<T>::define_copy_constructor() const
834
860
  requires std::copy_constructible<T>
835
861
  {
836
- auto const callback = detail::method_callback<arg::Self<Value>, arg::Arg<T const &>>::alloc(
862
+ auto const callback = detail::method_callback<args::Self<Value>, args::Arg<T const &>>::alloc(
837
863
  typed_data::DataType<T>::initialize_copy);
838
864
  detail::protect([&]() noexcept {
839
865
  using namespace literals;
@@ -0,0 +1,3 @@
1
+ require_relative '../mkmf'
2
+ include RCX::MakeMakefile
3
+ setup_rcx(cxx_standard: 'c++20')
data/lib/rcx/mkmf.rb CHANGED
@@ -1,56 +1,76 @@
1
1
  # SPDX-License-Identifier: BSL-1.0
2
2
  # SPDX-FileCopyrightText: Copyright 2024-2025 Kasumi Hanazuki <kasumi@rollingapple.net>
3
3
  require 'mkmf'
4
- include MakeMakefile['C++']
4
+ require_relative '../rcx'
5
5
 
6
- root = File.join(__dir__, '../..')
6
+ module RCX
7
+ CXX_STANDARD_FLAGS = {
8
+ 'c++20' => %w[--std=c++20 --std=c++2a].freeze,
9
+ 'c++23' => %w[--std=c++23 --std=c++2b].freeze,
10
+ 'c++26' => %w[--std=c++26 --std=c++2c].freeze,
11
+ }.freeze
7
12
 
8
- $INCFLAGS << " -I#{File.join(root, 'include').shellescape}"
13
+ root = File.join(__dir__, '../..')
14
+ INCDIR = File.join(root, 'include').shellescape
15
+ HEADERS = Dir[File.join(root, 'include/**/*.hpp')]
9
16
 
10
- $rcx_headers = Dir[File.join(root, 'include/**/*.hpp')]
17
+ module MakeMakefile
18
+ include ::MakeMakefile['C++']
11
19
 
12
- include (Module.new do
13
- def configuration(...)
14
- super.tap do |mk|
15
- mk << <<MAKEFILE
16
- rcx_headers = #{$rcx_headers.join(?\s)}
17
- ruby_headers := $(ruby_headers) $(rcx_headers)
18
- MAKEFILE
19
- end
20
- end
21
- end)
20
+ def setup_rcx(cxx_standard: 'c++20')
21
+ CXX_STANDARD_FLAGS.fetch(cxx_standard).find do |flag|
22
+ if checking_for("whether #{flag} is accepted as CXXFLAGS") { try_cflags(flag) }
23
+ $CXXFLAGS << " " << flag
24
+ true
25
+ else
26
+ false
27
+ end
28
+ end or raise "C++ compiler does not support #{cxx_standard}"
22
29
 
23
- ## libffi
30
+ $INCFLAGS << " -I#{INCDIR}"
24
31
 
25
- dir_config('libffi').any? || pkg_config('libffi')
26
- ffi_h = 'ffi.h'
27
- unless have_func('ffi_prep_cif', ffi_h)
28
- raise "libffi was not found"
29
- end
30
- unless have_func('ffi_closure_alloc', ffi_h) && have_func('ffi_prep_closure_loc', ffi_h)
31
- raise "libffi does not support closures"
32
- end
32
+ ## libffi
33
+ dir_config('libffi').any? || pkg_config('libffi')
34
+ ffi_h = 'ffi.h'
35
+ unless have_func('ffi_prep_cif', ffi_h)
36
+ raise "libffi was not found"
37
+ end
38
+ unless have_func('ffi_closure_alloc', ffi_h) && have_func('ffi_prep_closure_loc', ffi_h)
39
+ raise "libffi does not support closures"
40
+ end
33
41
 
34
- if have_header('cxxabi.h')
35
- have_func('abi::__cxa_demangle', 'cxxabi.h')
36
- have_func('abi::__cxa_current_exception_type', 'cxxabi.h')
37
- end
42
+ if have_header('cxxabi.h')
43
+ have_func('abi::__cxa_demangle', 'cxxabi.h')
44
+ have_func('abi::__cxa_current_exception_type', 'cxxabi.h')
45
+ end
38
46
 
39
- if checking_for("std::is_layout_compatible<>") {
40
- try_compile(<<'CXX')
47
+ if checking_for("std::is_layout_compatible<>") {
48
+ try_compile(<<'CXX')
41
49
  #include <type_traits>
42
50
  struct A { int a; };
43
51
  struct B { int b; };
44
52
  static_assert(std::is_layout_compatible<A, B>::value);
45
53
  CXX
46
- }
47
- $defs.push("-DHAVE_STD_IS_LAYOUT_COMPATIBLE=1")
48
- end
54
+ }
55
+ $defs.push("-DHAVE_STD_IS_LAYOUT_COMPATIBLE=1")
56
+ end
49
57
 
50
- if checking_for("nullability extension") {
51
- try_compile("void *_Nullable p, *_Nonnull q;")
52
- }
53
- $defs.push("-DHAVE_FEATURE_NULLABILITY=1")
54
- end
58
+ if checking_for("nullability extension") {
59
+ try_compile("void *_Nullable p, *_Nonnull q;")
60
+ }
61
+ $defs.push("-DHAVE_FEATURE_NULLABILITY=1")
62
+ end
55
63
 
56
- have_func('ruby_thread_has_gvl_p', 'ruby/thread.h')
64
+ have_func('ruby_thread_has_gvl_p', 'ruby/thread.h')
65
+ end
66
+
67
+ def configuration(...)
68
+ super.tap do |mk|
69
+ mk << <<MAKEFILE
70
+ rcx_headers = #{RCX::HEADERS.join(?\s)}
71
+ ruby_headers := $(ruby_headers) $(rcx_headers)
72
+ MAKEFILE
73
+ end
74
+ end
75
+ end
76
+ end
data/lib/rcx/version.rb CHANGED
@@ -1,3 +1,3 @@
1
1
  module RCX
2
- VERSION = -'0.3.0'
2
+ VERSION = -'0.4.0'
3
3
  end
metadata CHANGED
@@ -1,7 +1,7 @@
1
1
  --- !ruby/object:Gem::Specification
2
2
  name: rcx
3
3
  version: !ruby/object:Gem::Version
4
- version: 0.3.0
4
+ version: 0.4.0
5
5
  platform: ruby
6
6
  authors:
7
7
  - Kasumi Hanazuki
@@ -44,6 +44,7 @@ files:
44
44
  - include/rcx/rcx.hpp
45
45
  - lib/rcx.rb
46
46
  - lib/rcx/mkmf.rb
47
+ - lib/rcx/mkmf/c++20.rb
47
48
  - lib/rcx/version.rb
48
49
  - package.json
49
50
  - yarn.lock