Project

General

Profile

Actions

Feature #22128

open

C API: Expose RB_OBJ_SET_FROZEN_SHAREABLE

Feature #22128: C API: Expose RB_OBJ_SET_FROZEN_SHAREABLE

Added by byroot (Jean Boussier) 1 day ago. Updated about 4 hours ago.

Status:
Open
Assignee:
-
Target version:
-
[ruby-core:125830]

Description

Context

I'm trying to experiment with adapting Active Record for a Ractor architecture.

Since database connections can't possibly be Ractor shareable, the idea is to warp each connection inside its own ractor, and then send SQL queries and responses through a port.

But for this to perform well, I'd like to directly build the query response as a fully shareable object, so that it can be pushed into the port for free, instead of having Ruby need to recursively walk the potentially large response to mark objects as shareable.

Here's an example of how it would work in trilogy: https://github.com/trilogy-libraries/trilogy/pull/299

Problem

Unfortunately, the necessary API isn't currently exposed in the C API:

  • RB_OBJ_SET_FROZEN_SHAREABLE
  • RB_OBJ_SET_SHAREABLE / rb_obj_set_shareable

I understand that this API could potentially be misused, but given it's a C API, I believe it's acceptable to require care from the caller.

Benchmark

# frozen_string_literal: true

require 'trilogy'
require 'benchmark/ips'

baseline = Trilogy.new(database: "test")
shareable = Trilogy.new(database: "test", shareable: true)


values = {
  null_test: "test",
  bit_test: "test",
  single_bit_test: 1,
  tiny_int_test: 2,
  bool_cast_test: true,
  small_int_test: 4,
  medium_int_test: 23434,
  int_test: 324234,
  big_int_test: 234234,
  unsigned_big_int_test: 23423423,
  float_test: 234234,
  float_zero_test: 213.23,
  double_test: 23123.12323,
  decimal_test: 123213.12312,
  decimal_zero_test: 213213.21323,
  date_test: "2026-01-30",
  date_time_test: "2026-01-30 14:03:56",
  date_time_with_precision_test: "2026-01-30 14:03:56.12",
  time_with_precision_test: "2026-01-30 14:03:56.12",
  timestamp_test: "2026-01-30 14:03:56.12",
  varchar_test: "VARCHAR"
}

baseline.query("DELETE FROM trilogy_test")
insert = "INSERT INTO trilogy_test(#{values.keys.join(", ")}) VALUES (#{values.values.map(&:inspect).join(", ")})"
1000.times do
  baseline.query(insert)
end

p shareable.query("SELECT * FROM test.trilogy_test").to_a.size
Benchmark.ips do |x|
  x.report("baseline") { Ractor.make_shareable(baseline.query("SELECT * FROM trilogy_test")) }
  x.report("shareable") { Ractor.make_shareable(shareable.query("SELECT * FROM trilogy_test")) }
  x.compare!(order: :baseline)
end
ruby 4.1.0dev (2026-06-24T13:17:28Z expose-ractor-set-.. f9d7dd50cd) +PRISM [arm64-darwin25]
Warming up --------------------------------------
            baseline    42.000 i/100ms
           shareable    66.000 i/100ms
Calculating -------------------------------------
            baseline    220.024 (±43.2%) i/s    (4.54 ms/i) -      1.134k in   5.153976s
           shareable    677.487 (± 2.4%) i/s    (1.48 ms/i) -      3.432k in   5.065782s

Comparison:
 baseline:      220.0 i/s
shareable:      677.5 i/s - 3.08x  faster
Actions

Also available in: PDF Atom