numo-linalg 0.0.1

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.
Files changed (85) hide show
  1. checksums.yaml +7 -0
  2. data/Gemfile +4 -0
  3. data/README.md +80 -0
  4. data/Rakefile +18 -0
  5. data/ext/numo/linalg/blas/blas.c +352 -0
  6. data/ext/numo/linalg/blas/cblas.h +575 -0
  7. data/ext/numo/linalg/blas/cblas_t.h +563 -0
  8. data/ext/numo/linalg/blas/depend.erb +23 -0
  9. data/ext/numo/linalg/blas/extconf.rb +67 -0
  10. data/ext/numo/linalg/blas/gen/cogen.rb +72 -0
  11. data/ext/numo/linalg/blas/gen/decl.rb +203 -0
  12. data/ext/numo/linalg/blas/gen/desc.rb +8138 -0
  13. data/ext/numo/linalg/blas/gen/erbpp2.rb +339 -0
  14. data/ext/numo/linalg/blas/gen/replace_cblas_h.rb +27 -0
  15. data/ext/numo/linalg/blas/gen/spec.rb +93 -0
  16. data/ext/numo/linalg/blas/numo_blas.h +41 -0
  17. data/ext/numo/linalg/blas/tmpl/axpy.c +75 -0
  18. data/ext/numo/linalg/blas/tmpl/copy.c +57 -0
  19. data/ext/numo/linalg/blas/tmpl/def_c.c +3 -0
  20. data/ext/numo/linalg/blas/tmpl/def_d.c +3 -0
  21. data/ext/numo/linalg/blas/tmpl/def_s.c +3 -0
  22. data/ext/numo/linalg/blas/tmpl/def_z.c +3 -0
  23. data/ext/numo/linalg/blas/tmpl/dot.c +68 -0
  24. data/ext/numo/linalg/blas/tmpl/ger.c +114 -0
  25. data/ext/numo/linalg/blas/tmpl/init_class.c +20 -0
  26. data/ext/numo/linalg/blas/tmpl/init_module.c +12 -0
  27. data/ext/numo/linalg/blas/tmpl/lib.c +40 -0
  28. data/ext/numo/linalg/blas/tmpl/mm.c +214 -0
  29. data/ext/numo/linalg/blas/tmpl/module.c +9 -0
  30. data/ext/numo/linalg/blas/tmpl/mv.c +194 -0
  31. data/ext/numo/linalg/blas/tmpl/nrm2.c +79 -0
  32. data/ext/numo/linalg/blas/tmpl/rot.c +65 -0
  33. data/ext/numo/linalg/blas/tmpl/rotm.c +82 -0
  34. data/ext/numo/linalg/blas/tmpl/scal.c +69 -0
  35. data/ext/numo/linalg/blas/tmpl/sdsdot.c +77 -0
  36. data/ext/numo/linalg/blas/tmpl/set_prefix.c +16 -0
  37. data/ext/numo/linalg/blas/tmpl/swap.c +57 -0
  38. data/ext/numo/linalg/blas/tmpl/syr.c +102 -0
  39. data/ext/numo/linalg/blas/tmpl/syr2.c +110 -0
  40. data/ext/numo/linalg/blas/tmpl/syr2k.c +129 -0
  41. data/ext/numo/linalg/blas/tmpl/syrk.c +132 -0
  42. data/ext/numo/linalg/lapack/depend.erb +23 -0
  43. data/ext/numo/linalg/lapack/extconf.rb +45 -0
  44. data/ext/numo/linalg/lapack/gen/cogen.rb +74 -0
  45. data/ext/numo/linalg/lapack/gen/desc.rb +151278 -0
  46. data/ext/numo/linalg/lapack/gen/replace_lapacke_h.rb +32 -0
  47. data/ext/numo/linalg/lapack/gen/spec.rb +104 -0
  48. data/ext/numo/linalg/lapack/lapack.c +387 -0
  49. data/ext/numo/linalg/lapack/lapacke.h +16425 -0
  50. data/ext/numo/linalg/lapack/lapacke_config.h +119 -0
  51. data/ext/numo/linalg/lapack/lapacke_mangling.h +17 -0
  52. data/ext/numo/linalg/lapack/lapacke_t.h +10550 -0
  53. data/ext/numo/linalg/lapack/numo_lapack.h +42 -0
  54. data/ext/numo/linalg/lapack/tmpl/def_c.c +3 -0
  55. data/ext/numo/linalg/lapack/tmpl/def_d.c +7 -0
  56. data/ext/numo/linalg/lapack/tmpl/def_s.c +7 -0
  57. data/ext/numo/linalg/lapack/tmpl/def_z.c +3 -0
  58. data/ext/numo/linalg/lapack/tmpl/fact.c +179 -0
  59. data/ext/numo/linalg/lapack/tmpl/geev.c +123 -0
  60. data/ext/numo/linalg/lapack/tmpl/gels.c +232 -0
  61. data/ext/numo/linalg/lapack/tmpl/gesv.c +149 -0
  62. data/ext/numo/linalg/lapack/tmpl/gesvd.c +189 -0
  63. data/ext/numo/linalg/lapack/tmpl/ggev.c +138 -0
  64. data/ext/numo/linalg/lapack/tmpl/gqr.c +121 -0
  65. data/ext/numo/linalg/lapack/tmpl/init_class.c +20 -0
  66. data/ext/numo/linalg/lapack/tmpl/init_module.c +12 -0
  67. data/ext/numo/linalg/lapack/tmpl/lange.c +79 -0
  68. data/ext/numo/linalg/lapack/tmpl/lib.c +40 -0
  69. data/ext/numo/linalg/lapack/tmpl/module.c +9 -0
  70. data/ext/numo/linalg/lapack/tmpl/syev.c +91 -0
  71. data/ext/numo/linalg/lapack/tmpl/sygv.c +104 -0
  72. data/ext/numo/linalg/lapack/tmpl/trf.c +276 -0
  73. data/ext/numo/linalg/numo_linalg.h +115 -0
  74. data/lib/numo/linalg.rb +3 -0
  75. data/lib/numo/linalg/function.rb +1008 -0
  76. data/lib/numo/linalg/linalg.rb +7 -0
  77. data/lib/numo/linalg/loader.rb +174 -0
  78. data/lib/numo/linalg/use/atlas.rb +3 -0
  79. data/lib/numo/linalg/use/lapack.rb +3 -0
  80. data/lib/numo/linalg/use/mkl.rb +3 -0
  81. data/lib/numo/linalg/use/openblas.rb +3 -0
  82. data/lib/numo/linalg/version.rb +5 -0
  83. data/numo-linalg.gemspec +26 -0
  84. data/spec/lapack_spec.rb +13 -0
  85. metadata +172 -0
@@ -0,0 +1,121 @@
1
+ /*<%
2
+ %>*/
3
+ <% %>
4
+ #define args_t <%=func_name%>_args_t
5
+ #define func_p <%=func_name%>_p
6
+
7
+ typedef struct {
8
+ int order;
9
+ } args_t;
10
+
11
+ static <%=func_name%>_t func_p = 0;
12
+
13
+ static void
14
+ <%=c_iter%>(na_loop_t * const lp)
15
+ {
16
+ dtype *a;
17
+ dtype *tau;
18
+ int *info;
19
+ int m, n, k, lda, tmp;
20
+ args_t *g;
21
+
22
+ a = (dtype*)NDL_PTR(lp,0);
23
+ tau = (dtype*)NDL_PTR(lp,1);
24
+ info = (int*)NDL_PTR(lp,2);
25
+ g = (args_t*)(lp->opt_ptr);
26
+
27
+ m = NDL_SHAPE(lp,0)[0];
28
+ n = NDL_SHAPE(lp,0)[1];
29
+ k = NDL_SHAPE(lp,1)[0];
30
+ SWAP_IFCOL(g->order,m,n);
31
+ lda = NDL_STEP(lp,0) / sizeof(dtype);
32
+
33
+ printf("order=%d m=%d n=%d k=%d lda=%d \n",g->order,m,n,k,lda);
34
+
35
+ *info = (*func_p)(g->order, m, n, k, a, lda, tau);
36
+ CHECK_ERROR(*info);
37
+ }
38
+
39
+ /*<%
40
+ args_v = [
41
+ "a",
42
+ "tau",
43
+ ].select{|x| x}.join(", ")
44
+
45
+ args_opt = [
46
+ "order:'R'",
47
+ ].select{|x| x}.join(", ")
48
+
49
+ params = [
50
+ mat("a",:inplace),
51
+ vec("tau"),
52
+ opt("order"),
53
+ ].select{|x| x}.join("\n ")
54
+
55
+ return_type = [
56
+ class_name,
57
+ "Integer"
58
+ ].select{|x| x}.join(", ")
59
+
60
+ return_name = [
61
+ "a",
62
+ "info"
63
+ ].select{|x| x}.join(", ")
64
+ %>
65
+ @overload <%=name%>(a, tau, order:'R')
66
+ <%=params%>
67
+ @return [[<%=return_name%>]] Array<<%=return_type%>>
68
+ <%=outparam(return_name)%>
69
+
70
+ <%=description%>
71
+
72
+ */
73
+ static VALUE
74
+ <%=c_func(-1)%>(int argc, VALUE const argv[], VALUE UNUSED(mod))
75
+ {
76
+ VALUE a, tau, ans;
77
+ int m, n, k, tmp;
78
+ narray_t *na1, *na2;
79
+ ndfunc_arg_in_t ain[2] = {{OVERWRITE,2},{cT,1}};
80
+ ndfunc_arg_out_t aout[1] = {{cInt,0}};
81
+ ndfunc_t ndf = {&<%=c_iter%>, NO_LOOP|NDF_EXTRACT, 2,1, ain,aout};
82
+
83
+ args_t g = {0};
84
+ VALUE opts[1] = {Qundef};
85
+ VALUE kw_hash = Qnil;
86
+ ID kw_table[1] = {id_order};
87
+
88
+ CHECK_FUNC(func_p,"<%=func_name%>");
89
+
90
+ rb_scan_args(argc, argv, "2:", &a, &tau, &kw_hash);
91
+ rb_get_kwargs(kw_hash, kw_table, 0, 1, opts);
92
+ g.order = option_order(opts[0]);
93
+
94
+ COPY_OR_CAST_TO(a,cT);
95
+ GetNArray(a, na1);
96
+ CHECK_DIM_GE(na1, 2);
97
+ m = ROW_SIZE(na1);
98
+ n = COL_SIZE(na1);
99
+
100
+ GetNArray(tau, na2);
101
+ CHECK_DIM_GE(na2, 1);
102
+ k = COL_SIZE(na2);
103
+ if (m < n) {
104
+ rb_raise(nary_eShapeError,
105
+ "a row length (m) must be >= a column length (n): m=%d n=%d",
106
+ m,n);
107
+ }
108
+ if (n < k) {
109
+ rb_raise(nary_eShapeError,
110
+ "a column length (n) must be >= tau length (k): n=%d, k=%d",
111
+ k,n);
112
+ }
113
+ SWAP_IFCOL(g.order,m,n);
114
+
115
+ ans = na_ndloop3(&ndf, &g, 2, a, tau);
116
+
117
+ return rb_assoc_new(a, ans);
118
+ }
119
+
120
+ #undef args_t
121
+ #undef func_p
@@ -0,0 +1,20 @@
1
+ /*
2
+ Document-class: <%= full_class_name %>
3
+ <%= description %>
4
+ */
5
+ cT = rb_define_class_under(<%=ns_var%>, "<%=class_name%>", cNArray);
6
+
7
+ <% for x in class_alias %>
8
+ // alias of <%=class_name%>
9
+ rb_define_const(<%=ns_var%>, "<%=x%>", <%=type_var%>);
10
+ <% end %>
11
+
12
+ hCast = rb_hash_new();
13
+ rb_define_const(cT, "UPCAST", hCast);
14
+ rb_hash_aset(hCast, rb_cArray, cT);
15
+ <% for x in upcast %>
16
+ <%= x %><% end %>
17
+
18
+ <% @children.each do |m| %>
19
+ <%= m.init_def %><% end %>
20
+ rb_define_singleton_method(cT, "[]", <%=find("cast").c_func%>, -2);
@@ -0,0 +1,12 @@
1
+ /*
2
+ Document-module: <%= full_module_name %>
3
+ <%= description %>
4
+ */
5
+ <% if module_var != ns_var %>
6
+ <%=module_var%> = rb_define_module_under(<%=ns_var%>, "<%=module_name%>");
7
+ <% end %>
8
+ <% @children.each do |m| %>
9
+ <%= m.init_def %><% end %>
10
+
11
+ // how to do this?
12
+ //rb_extend_object(cT, mTM);
@@ -0,0 +1,79 @@
1
+ #define args_t <%=func_name%>_args_t
2
+ #define func_p <%=func_name%>_p
3
+
4
+ typedef struct {
5
+ int order;
6
+ char norm;
7
+ } args_t;
8
+
9
+ static <%=func_name%>_t func_p = 0;
10
+
11
+ static void
12
+ <%=c_iter%>(na_loop_t * const lp)
13
+ {
14
+ dtype *a;
15
+ rtype *norm;
16
+ int m, n, lda, tmp;
17
+ args_t *g;
18
+
19
+ a = (dtype*)NDL_PTR(lp,0);
20
+ norm = (rtype*)NDL_PTR(lp,1);
21
+ g = (args_t*)(lp->opt_ptr);
22
+
23
+ m = NDL_SHAPE(lp,0)[0];
24
+ n = NDL_SHAPE(lp,0)[1];
25
+ SWAP_IFCOL(g->order,m,n);
26
+ lda = NDL_STEP(lp,0) / sizeof(dtype);
27
+
28
+ //printf("order=%d norm=%c m=%d n=%d lda=%d\n",g->order,g->norm,m,n,lda);
29
+
30
+ *norm = (*func_p)(g->order, g->norm, m, n, a, lda);
31
+ }
32
+
33
+ /*<%
34
+ params = [
35
+ mat("a"),
36
+ "@param [String] norm Kind of norm: 'M',('1','O'),'I',('F','E')",
37
+ opt("order"),
38
+ ].select{|x| x}.join("\n ")
39
+ %>
40
+ @overload <%=name%>(a, norm, [order:'R'])
41
+ <%=params%>
42
+ @return [<%=real_class_name%>] returns <%=name%>.
43
+
44
+ <%=description%>
45
+
46
+ */
47
+ static VALUE
48
+ <%=c_func(-1)%>(int argc, VALUE const argv[], VALUE UNUSED(mod))
49
+ {
50
+ VALUE a, norm, ans;
51
+ narray_t *na1;
52
+ ndfunc_arg_in_t ain[1] = {{cT,2}};
53
+ ndfunc_arg_out_t aout[1] = {{cRT,0}};
54
+ ndfunc_t ndf = {&<%=c_iter%>, NO_LOOP|NDF_EXTRACT, 1, 1, ain, aout};
55
+
56
+ args_t g;
57
+ VALUE opts[1] = {Qundef};
58
+ ID kw_table[1] = {id_order};
59
+ VALUE kw_hash = Qnil;
60
+
61
+ CHECK_FUNC(func_p,"<%=func_name%>");
62
+
63
+ rb_scan_args(argc, argv, "2:", &a, &norm, &kw_hash);
64
+ rb_get_kwargs(kw_hash, kw_table, 0, 1, opts);
65
+ g.order = option_order(opts[0]);
66
+ g.norm = option_job(norm,'F','F');
67
+ //reduce = nary_reduce_options(Qnil, &opts[1], 1, &a, &ndf);
68
+ //A is DOUBLE PRECISION array, dimension (LDA,N)
69
+ //On entry, the M-by-N matrix A.
70
+ //COPY_OR_CAST_TO(a,cT); // not overwrite
71
+ GetNArray(a, na1);
72
+ CHECK_DIM_GE(na1, 2);
73
+
74
+ ans = na_ndloop3(&ndf, &g, 1, a);
75
+ return ans;
76
+ }
77
+
78
+ #undef func_p
79
+ #undef args_t
@@ -0,0 +1,40 @@
1
+ /*
2
+ LAPACK wrapper for Ruby/Numo
3
+ (C) Copyright 2017 by Masahiro TANAKA
4
+
5
+ This program is free software.
6
+ NO WARRANTY.
7
+ */
8
+
9
+ #include <assert.h>
10
+ #include <ruby.h>
11
+ #include "numo/narray.h"
12
+ #include "numo/template.h"
13
+ #include "numo_lapack.h"
14
+
15
+ <% id_decl.each do |x| %>
16
+ <%= x %>
17
+ <% end %>
18
+
19
+ <% include_files.each do |f| %>
20
+ #include <<%=f%>>
21
+ <% end %>
22
+
23
+ <% children.each do |c|%>
24
+ <%= c.result+"\n\n" %>
25
+ <% end %>
26
+
27
+ void
28
+ Init_<%=lib_name%>(void)
29
+ {
30
+ VALUE <%=ns_var%>;
31
+
32
+ <%=ns_var%> = rb_define_module("Numo");
33
+
34
+ <% id_assign.each do |x| %>
35
+ <%= x %><% end %>
36
+
37
+ <% children.each do |c| %>
38
+ <%= c.init_def %>
39
+ <% end %>
40
+ }
@@ -0,0 +1,9 @@
1
+ /*
2
+ module definition: <%= full_module_name %>
3
+ */
4
+
5
+ <% if module_var != ns_var %>
6
+ static VALUE <%=module_var%>;
7
+ <% end %>
8
+
9
+ <%= method_code %>
@@ -0,0 +1,91 @@
1
+ #define args_t <%=func_name%>_args_t
2
+ #define func_p <%=func_name%>_p
3
+
4
+ typedef struct {
5
+ int order;
6
+ char jobz, uplo;
7
+ } args_t;
8
+
9
+ static <%=func_name%>_t func_p = 0;
10
+
11
+ static void
12
+ <%=c_iter%>(na_loop_t * const lp)
13
+ {
14
+ dtype *a;
15
+ rtype *w;
16
+ int *info;
17
+ int n, lda;
18
+ args_t *g;
19
+
20
+ a = (dtype*)NDL_PTR(lp,0);
21
+ w = (rtype*)NDL_PTR(lp,1);
22
+ info = (int*)NDL_PTR(lp,2);
23
+ g = (args_t*)(lp->opt_ptr);
24
+
25
+ n = NDL_SHAPE(lp,0)[1];
26
+ lda = NDL_STEP(lp,0) / sizeof(dtype);
27
+
28
+ //printf("order=%d jobz=%c uplo=%c n=%d lda=%d\n",g->order,g->jobz,g->uplo,n,lda);
29
+
30
+ *info = (*func_p)( g->order, g->jobz, g->uplo, n, a, lda, w );
31
+ CHECK_ERROR(*info);
32
+ }
33
+
34
+ /*<%
35
+ params = [
36
+ mat("a",:inplace),
37
+ jobe("jobz"),
38
+ opt("uplo"),
39
+ opt("order"),
40
+ ].select{|x| x}.join("\n ")
41
+ return_name = "a, w, info"
42
+ %>
43
+ @overload <%=name%>(a, [jobz:'V', uplo:'U', order:'R'])
44
+ <%=params%>
45
+ @return [[<%=return_name%>]] Array<<%=real_class_name%>,<%=real_class_name%>,Integer>
46
+ <%=outparam(return_name)%>
47
+
48
+ <%=description%>
49
+
50
+ */
51
+ static VALUE
52
+ <%=c_func(-1)%>(int argc, VALUE const argv[], VALUE UNUSED(mod))
53
+ {
54
+ VALUE a, ans;
55
+ int m, n;
56
+ narray_t *na1;
57
+ size_t shape[1];
58
+ ndfunc_arg_in_t ain[1] = {{OVERWRITE,2}};
59
+ ndfunc_arg_out_t aout[2] = {{cRT,1,shape},{cInt,0}};
60
+ ndfunc_t ndf = {&<%=c_iter%>, NO_LOOP|NDF_EXTRACT, 1, 2, ain, aout};
61
+
62
+ args_t g = {0,0,0};
63
+ VALUE opts[3] = {Qundef,Qundef,Qundef};
64
+ VALUE kw_hash = Qnil;
65
+ ID kw_table[3] = {id_order,id_jobz,id_uplo};
66
+
67
+ CHECK_FUNC(func_p,"<%=func_name%>");
68
+
69
+ rb_scan_args(argc, argv, "1:", &a, &kw_hash);
70
+ rb_get_kwargs(kw_hash, kw_table, 0, 3, opts);
71
+ g.order = option_order(opts[0]);
72
+ g.jobz = option_job(opts[1],'V','N');
73
+ g.uplo = option_uplo(opts[2]);
74
+
75
+ COPY_OR_CAST_TO(a,cT);
76
+ GetNArray(a, na1);
77
+ CHECK_DIM_GE(na1, 2);
78
+ m = ROW_SIZE(na1);
79
+ n = COL_SIZE(na1);
80
+ if (m != n) {
81
+ rb_raise(nary_eShapeError,"matrix must be square");
82
+ }
83
+ shape[0] = n;
84
+
85
+ ans = na_ndloop3(&ndf, &g, 1, a);
86
+
87
+ return rb_ary_new3(3, a, RARRAY_AREF(ans,0), RARRAY_AREF(ans,1));
88
+ }
89
+
90
+ #undef args_t
91
+ #undef func_p
@@ -0,0 +1,104 @@
1
+ #define args_t <%=func_name%>_args_t
2
+ #define func_p <%=func_name%>_p
3
+
4
+ typedef struct {
5
+ int order;
6
+ int itype;
7
+ char jobz, uplo;
8
+ } args_t;
9
+
10
+ static <%=func_name%>_t func_p = 0;
11
+
12
+ static void
13
+ <%=c_iter%>(na_loop_t * const lp)
14
+ {
15
+ dtype *a, *b;
16
+ rtype *w;
17
+ int *info;
18
+ int n, lda, ldb;
19
+ args_t *g;
20
+
21
+ a = (dtype*)NDL_PTR(lp,0);
22
+ b = (dtype*)NDL_PTR(lp,1);
23
+ w = (rtype*)NDL_PTR(lp,2);
24
+ info = (int*)NDL_PTR(lp,3);
25
+ g = (args_t*)(lp->opt_ptr);
26
+
27
+ n = NDL_SHAPE(lp,0)[1];
28
+ lda = NDL_STEP(lp,0) / sizeof(dtype);
29
+ ldb = NDL_STEP(lp,1) / sizeof(dtype);
30
+
31
+ //printf("order=%d itype=%d jobz=%c uplo=%c n=%d lda=%d ldb=%d\n",g->order,g->itype,g->jobz,g->uplo,n,lda,ldb);
32
+
33
+ *info = (*func_p)( g->order, g->itype, g->jobz, g->uplo, n, a, lda, b, ldb, w );
34
+ CHECK_ERROR(*info);
35
+ }
36
+
37
+ /*<%
38
+ params = [
39
+ mat("a",:inplace),
40
+ mat("b",:inplace),
41
+ "@param [Integer] itype Specifies the problem type to be solved. If 1: A*x = (lambda)*B*x, If 2: A*B*x = (lambda)*x, If 3: B*A*x = (lambda)*x.",
42
+ jobe("jobz"),
43
+ opt("uplo"),
44
+ opt("order"),
45
+ ].select{|x| x}.join("\n ")
46
+ return_name="a, b, w, info"
47
+ %>
48
+ @overload <%=name%>(a, b, [itype:1, jobz:'V', uplo:'U', order:'R'])
49
+ <%=params%>
50
+ @return [[<%=return_name%>]] Array<<%=real_class_name%>,<%=real_class_name%>,Integer>
51
+ <%=outparam(return_name)%>
52
+
53
+ <%=description%>
54
+
55
+ */
56
+ static VALUE
57
+ <%=c_func(-1)%>(int argc, VALUE const argv[], VALUE UNUSED(mod))
58
+ {
59
+ VALUE a, b, ans;
60
+ int n, nb;
61
+ narray_t *na1, *na2;
62
+ size_t shape[1];
63
+ ndfunc_arg_in_t ain[2] = {{OVERWRITE,2},{OVERWRITE,2}};
64
+ ndfunc_arg_out_t aout[2] = {{cRT,1,shape},{cInt,0}};
65
+ ndfunc_t ndf = {&<%=c_iter%>, NO_LOOP|NDF_EXTRACT, 2, 2, ain, aout};
66
+
67
+ args_t g;
68
+ VALUE opts[4] = {Qundef,Qundef,Qundef,Qundef};
69
+ VALUE kw_hash = Qnil;
70
+ ID kw_table[4] = {id_order,id_jobz,id_uplo,id_itype};
71
+
72
+ CHECK_FUNC(func_p,"<%=func_name%>");
73
+
74
+ rb_scan_args(argc, argv, "2:", &a, &b, &kw_hash);
75
+ rb_get_kwargs(kw_hash, kw_table, 0, 4, opts);
76
+ g.order = option_order(opts[0]);
77
+ g.jobz = option_job(opts[1],'V','N');
78
+ g.uplo = option_uplo(opts[2]);
79
+ g.itype = NUM2INT(option_value(opts[3],INT2FIX(1)));
80
+
81
+ COPY_OR_CAST_TO(a,cT);
82
+ GetNArray(a, na1);
83
+ CHECK_DIM_GE(na1, 2);
84
+
85
+ COPY_OR_CAST_TO(b,cT);
86
+ GetNArray(b, na2);
87
+ CHECK_DIM_GE(na2, 2);
88
+
89
+ CHECK_SQUARE("matrix a",na1);
90
+ n = COL_SIZE(na1);
91
+ CHECK_SQUARE("matrix b",na2);
92
+ nb = COL_SIZE(na2);
93
+ if (n != nb) {
94
+ rb_raise(nary_eShapeError,"matrix a and b must have same size");
95
+ }
96
+ shape[0] = n;
97
+
98
+ ans = na_ndloop3(&ndf, &g, 2, a, b);
99
+
100
+ return rb_ary_new3(4, a, b, RARRAY_AREF(ans,0), RARRAY_AREF(ans,1));
101
+ }
102
+
103
+ #undef args_t
104
+ #undef func_p