zig_example 0.2.0 → 0.3.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: 78cf562dbd8af1b7d9badd8701f811d23c53566e0533fd85d708d7aa8aacfae8
4
- data.tar.gz: 16379b58200d91656978a9ec696efc9f13b0b4b43ee4be590efe82504cfc4d29
3
+ metadata.gz: c4f811b0c8ce05fcc84a9094411461dad264023686f41fd9e536d2a82beaa5a6
4
+ data.tar.gz: a9aed665e0b482b12828d2eeaf6b5b82307c0e9af9006953d1e811931d348205
5
5
  SHA512:
6
- metadata.gz: e3130b68fb94dbbe6de189f122c9bf3b43be22a8fdff6fa827604cd081608a96e88252eb051f39669a7b216c69ed0c460b79ba1a85508f86887d9f3b653c1f96
7
- data.tar.gz: 8ca455621357daafa0f059f4137e4370bad1c813367bb13e86a9d9c658fa9a131bf105dd3699d76190d1c2a85d4c1eb2e316491142f0734d44c2485f07146ae9
6
+ metadata.gz: 31bdb7f97d43ec219621ea2fde263fd7e3647faf9b70114bf0c82f81859790233b8c2fd6fff3064070a501ac591bcb6f49ab6016d59582c9bb7ac3f065827af5
7
+ data.tar.gz: 4b22bcd639585929ed3c8d7de6a29af737bcebd4b0e00d8653a0830d22a87ffb4d6576f00dfdcb3f1b460ba0c7c5db91a1387ed4edbf4cf24da2bc3c7b44f4f5
@@ -13,7 +13,7 @@ File.open('Makefile', 'a') do |f|
13
13
  \t-$(Q)$(RM) -rf zig-cache zig-out
14
14
 
15
15
  #{TARGET}.so: Makefile build.zig src/main.zig
16
- \tenv -u DESTDIR RUBYLIBDIR=$(rubylibdir) RUBYHDRDIR=$(rubyhdrdir) RUBYARCHHDRDIR=$(rubyarchhdrdir) zig build test install
16
+ \tenv -u DESTDIR RUBYLIBDIR=$(rubylibdir) RUBYHDRDIR=$(rubyhdrdir) RUBYARCHHDRDIR=$(rubyarchhdrdir) zig build -Doptimize=ReleaseSafe test install
17
17
  \tcp -v zig-out/lib/#{TARGET}.so .
18
18
 
19
19
  MFILE
@@ -13,7 +13,7 @@ File.open('Makefile', 'a') do |f|
13
13
  \t-$(Q)$(RM) -rf zig-cache zig-out
14
14
 
15
15
  #{TARGET}.so: Makefile build.zig src/main.zig
16
- \tenv -u DESTDIR RUBYLIBDIR=$(rubylibdir) RUBYHDRDIR=$(rubyhdrdir) RUBYARCHHDRDIR=$(rubyarchhdrdir) zig build test install
16
+ \tenv -u DESTDIR RUBYLIBDIR=$(rubylibdir) RUBYHDRDIR=$(rubyhdrdir) RUBYARCHHDRDIR=$(rubyarchhdrdir) zig build -Doptimize=ReleaseSafe test install
17
17
  \tcp -v zig-out/lib/#{TARGET}.so .
18
18
 
19
19
  MFILE
@@ -0,0 +1,65 @@
1
+ const std = @import("std");
2
+
3
+ pub fn build(b: *std.Build) void {
4
+ const optimize = b.standardOptimizeOption(.{});
5
+ const target = b.standardTargetOptions(.{});
6
+
7
+ var rubylibdir = std.os.getenv("RUBYLIBDIR") orelse "";
8
+ var rubyhdrdir = std.os.getenv("RUBYHDRDIR") orelse "";
9
+ var rubyarchhdrdir = std.os.getenv("RUBYARCHHDRDIR") orelse "";
10
+
11
+ ////////////////////////////////////////////////////////////////
12
+ // lib
13
+
14
+ const clib = b.addStaticLibrary(.{
15
+ .name = "zigrb_lucas_lehmer",
16
+ .root_source_file = .{ .path = "src/lucas_lehmer.c" },
17
+ .target = target,
18
+ .optimize = optimize,
19
+ });
20
+ clib.force_pic = true;
21
+
22
+ clib.linkSystemLibrary("gmp");
23
+
24
+ const lib = b.addSharedLibrary(.{
25
+ .name = "zigrb_lucas_lehmer",
26
+ .root_source_file = .{ .path = "src/wrapper.zig" },
27
+ .target = target,
28
+ .optimize = optimize,
29
+ });
30
+
31
+ lib.addLibraryPath(rubylibdir);
32
+ lib.addIncludePath(rubyhdrdir);
33
+ lib.addIncludePath(rubyarchhdrdir);
34
+
35
+ lib.linkSystemLibrary("ruby");
36
+ lib.linkSystemLibrary("c");
37
+
38
+ lib.addIncludePath("src");
39
+ lib.linkLibrary(clib);
40
+
41
+ b.installArtifact(lib);
42
+
43
+ ////////////////////////////////////////////////////////////////
44
+ // test
45
+
46
+ const unit_tests = b.addTest(.{
47
+ .root_source_file = .{ .path = "src/wrapper.zig" },
48
+ .target = target,
49
+ .optimize = optimize,
50
+ });
51
+
52
+ unit_tests.addLibraryPath(rubylibdir);
53
+ unit_tests.addIncludePath(rubyhdrdir);
54
+ unit_tests.addIncludePath(rubyarchhdrdir);
55
+
56
+ unit_tests.linkSystemLibrary("ruby");
57
+ unit_tests.linkSystemLibrary("c");
58
+
59
+ unit_tests.addIncludePath("src");
60
+ unit_tests.linkLibrary(clib);
61
+
62
+ const run_unit_tests = b.addRunArtifact(unit_tests);
63
+ const test_step = b.step("test", "Run unit tests");
64
+ test_step.dependOn(&run_unit_tests.step);
65
+ }
@@ -0,0 +1,20 @@
1
+ require 'mkmf'
2
+
3
+ $objs = ['']
4
+ $srcs = ['']
5
+ TARGET = 'libzigrb_lucas_lehmer'
6
+
7
+ create_makefile TARGET
8
+ File.open('Makefile', 'a') do |f|
9
+ f.puts <<~MFILE
10
+
11
+ clean: clean-zig
12
+ clean-zig:
13
+ \t-$(Q)$(RM) -rf zig-cache zig-out
14
+
15
+ #{TARGET}.so: Makefile build.zig src/*
16
+ \tenv -u DESTDIR RUBYLIBDIR=$(rubylibdir) RUBYHDRDIR=$(rubyhdrdir) RUBYARCHHDRDIR=$(rubyarchhdrdir) zig build -Doptimize=ReleaseSafe test install
17
+ \tcp -v zig-out/lib/#{TARGET}.so .
18
+
19
+ MFILE
20
+ end
@@ -0,0 +1,60 @@
1
+ #include <gmp.h>
2
+
3
+ int lucas_lehmer(unsigned long p)
4
+ {
5
+ mpz_t V, mp, t;
6
+ unsigned long k, tlim;
7
+ int res;
8
+
9
+ if (p == 2) return 1;
10
+ if (!(p&1)) return 0;
11
+
12
+ mpz_init_set_ui(t, p);
13
+ if (!mpz_probab_prime_p(t, 25)) /* if p is composite, 2^p-1 is not prime */
14
+ { mpz_clear(t); return 0; }
15
+
16
+ if (p < 23) /* trust the PRP test for these values */
17
+ { mpz_clear(t); return (p != 11); }
18
+
19
+ mpz_init(mp);
20
+ mpz_setbit(mp, p);
21
+ mpz_sub_ui(mp, mp, 1);
22
+
23
+ /* If p=3 mod 4 and p,2p+1 both prime, then 2p+1 | 2^p-1. Cheap test. */
24
+ if (p > 3 && p % 4 == 3) {
25
+ mpz_mul_ui(t, t, 2);
26
+ mpz_add_ui(t, t, 1);
27
+ if (mpz_probab_prime_p(t,25) && mpz_divisible_p(mp, t))
28
+ { mpz_clear(mp); mpz_clear(t); return 0; }
29
+ }
30
+
31
+ /* Do a little trial division first. Saves quite a bit of time. */
32
+ tlim = p/2;
33
+ if (tlim > (ULONG_MAX/(2*p)))
34
+ tlim = ULONG_MAX/(2*p);
35
+ for (k = 1; k < tlim; k++) {
36
+ unsigned long q = 2*p*k+1;
37
+ /* factor must be 1 or 7 mod 8 and a prime */
38
+ if ( (q%8==1 || q%8==7) &&
39
+ q % 3 && q % 5 && q % 7 &&
40
+ mpz_divisible_ui_p(mp, q) )
41
+ { mpz_clear(mp); mpz_clear(t); return 0; }
42
+ }
43
+
44
+ mpz_init_set_ui(V, 4);
45
+ for (k = 3; k <= p; k++) {
46
+ mpz_mul(V, V, V);
47
+ mpz_sub_ui(V, V, 2);
48
+ /* mpz_mod(V, V, mp) but more efficiently done given mod 2^p-1 */
49
+ if (mpz_sgn(V) < 0) mpz_add(V, V, mp);
50
+ /* while (n > mp) { n = (n >> p) + (n & mp) } if (n==mp) n=0 */
51
+ /* but in this case we can have at most one loop plus a carry */
52
+ mpz_tdiv_r_2exp(t, V, p);
53
+ mpz_tdiv_q_2exp(V, V, p);
54
+ mpz_add(V, V, t);
55
+ while (mpz_cmp(V, mp) >= 0) mpz_sub(V, V, mp);
56
+ }
57
+ res = !mpz_sgn(V);
58
+ mpz_clear(t); mpz_clear(mp); mpz_clear(V);
59
+ return res;
60
+ }
@@ -0,0 +1 @@
1
+ int lucas_lehmer(unsigned long p);
@@ -0,0 +1,42 @@
1
+ const std = @import("std");
2
+ const testing = std.testing;
3
+
4
+ const ruby = @cImport(@cInclude("ruby/ruby.h"));
5
+ const clib = @cImport(@cInclude("lucas_lehmer.h"));
6
+
7
+ fn rb_lucas_lehmer(...) callconv(.C) ruby.VALUE {
8
+ var ap = @cVaStart();
9
+ defer @cVaEnd(&ap);
10
+
11
+ // first argument is `self` in ruby; discard it
12
+ var self = @cVaArg(&ap, ruby.VALUE); _ = self;
13
+
14
+ // back and forth type conversions + delegation
15
+ var p = ruby.NUM2ULONG(@cVaArg(&ap, ruby.VALUE));
16
+ return if (clib.lucas_lehmer(p) == 1) ruby.Qtrue else ruby.Qfalse;
17
+ }
18
+
19
+ export fn Init_libzigrb_lucas_lehmer() void {
20
+ ruby.ruby_init();
21
+ var zig_rb_class: ruby.VALUE = ruby.rb_define_class("ZigExample", ruby.rb_cObject);
22
+ _ = ruby.rb_define_method(zig_rb_class, "lucas_lehmer", rb_lucas_lehmer, 1);
23
+ }
24
+
25
+ test "lucas_lehmer 1 passes" {
26
+ try testing.expect(clib.lucas_lehmer(1) == 0);
27
+ }
28
+ test "lucas_lehmer 2 passes" {
29
+ try testing.expect(clib.lucas_lehmer(2) == 1);
30
+ }
31
+ test "lucas_lehmer 3 passes" {
32
+ try testing.expect(clib.lucas_lehmer(3) == 1);
33
+ }
34
+ test "lucas_lehmer 4 passes" {
35
+ try testing.expect(clib.lucas_lehmer(4) == 0);
36
+ }
37
+ test "lucas_lehmer 5 passes" {
38
+ try testing.expect(clib.lucas_lehmer(5) == 1);
39
+ }
40
+ test "lucas_lehmer 11 passes" {
41
+ try testing.expect(clib.lucas_lehmer(11) == 0);
42
+ }
@@ -1,5 +1,5 @@
1
1
  # frozen_string_literal: true
2
2
 
3
3
  class ZigExample
4
- VERSION = '0.2.0'
4
+ VERSION = '0.3.0'
5
5
  end
data/lib/zig_example.rb CHANGED
@@ -3,6 +3,7 @@ require_relative 'zig_example/version'
3
3
  %w{
4
4
  zigrb_100doors
5
5
  zigrb_ackermann
6
+ zigrb_lucas_lehmer
6
7
  }.each do |lib|
7
8
  begin
8
9
  require "lib#{lib}"
metadata CHANGED
@@ -1,14 +1,14 @@
1
1
  --- !ruby/object:Gem::Specification
2
2
  name: zig_example
3
3
  version: !ruby/object:Gem::Version
4
- version: 0.2.0
4
+ version: 0.3.0
5
5
  platform: ruby
6
6
  authors:
7
7
  - Frank J. Cameron
8
8
  autorequire:
9
9
  bindir: bin
10
10
  cert_chain: []
11
- date: 2023-04-21 00:00:00.000000000 Z
11
+ date: 2023-04-24 00:00:00.000000000 Z
12
12
  dependencies: []
13
13
  description:
14
14
  email:
@@ -17,6 +17,7 @@ executables: []
17
17
  extensions:
18
18
  - ext/zigrb_100doors/extconf.rb
19
19
  - ext/zigrb_ackermann/extconf.rb
20
+ - ext/zigrb_lucas_lehmer/extconf.rb
20
21
  extra_rdoc_files: []
21
22
  files:
22
23
  - ext/zigrb_100doors/build.zig
@@ -25,6 +26,11 @@ files:
25
26
  - ext/zigrb_ackermann/build.zig
26
27
  - ext/zigrb_ackermann/extconf.rb
27
28
  - ext/zigrb_ackermann/src/main.zig
29
+ - ext/zigrb_lucas_lehmer/build.zig
30
+ - ext/zigrb_lucas_lehmer/extconf.rb
31
+ - ext/zigrb_lucas_lehmer/src/lucas_lehmer.c
32
+ - ext/zigrb_lucas_lehmer/src/lucas_lehmer.h
33
+ - ext/zigrb_lucas_lehmer/src/wrapper.zig
28
34
  - lib/zig_example.rb
29
35
  - lib/zig_example/version.rb
30
36
  homepage: https://gitlab.com/fjc/ruby-zig_example
@@ -51,5 +57,5 @@ requirements: []
51
57
  rubygems_version: 3.2.5
52
58
  signing_key:
53
59
  specification_version: 4
54
- summary: An example gem for ruby with a native zig extension.
60
+ summary: An example gem for ruby with native zig extensions.
55
61
  test_files: []