nmatrix 0.0.5 → 0.0.6
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 +4 -4
- data/History.txt +102 -10
- data/README.rdoc +24 -32
- data/Rakefile +1 -1
- data/ext/nmatrix/data/complex.h +9 -0
- data/ext/nmatrix/data/data.cpp +78 -4
- data/ext/nmatrix/data/data.h +86 -54
- data/ext/nmatrix/data/rational.h +2 -0
- data/ext/nmatrix/data/ruby_object.h +38 -8
- data/ext/nmatrix/extconf.rb +13 -7
- data/ext/nmatrix/nmatrix.cpp +262 -139
- data/ext/nmatrix/nmatrix.h +11 -4
- data/ext/nmatrix/storage/common.cpp +20 -13
- data/ext/nmatrix/storage/common.h +18 -12
- data/ext/nmatrix/storage/dense.cpp +122 -192
- data/ext/nmatrix/storage/dense.h +4 -2
- data/ext/nmatrix/storage/list.cpp +467 -636
- data/ext/nmatrix/storage/list.h +6 -3
- data/ext/nmatrix/storage/storage.cpp +83 -46
- data/ext/nmatrix/storage/storage.h +7 -7
- data/ext/nmatrix/storage/yale.cpp +621 -361
- data/ext/nmatrix/storage/yale.h +21 -9
- data/ext/nmatrix/ttable_helper.rb +27 -31
- data/ext/nmatrix/types.h +1 -1
- data/ext/nmatrix/util/math.cpp +9 -10
- data/ext/nmatrix/util/sl_list.cpp +1 -7
- data/ext/nmatrix/util/sl_list.h +0 -118
- data/lib/nmatrix/blas.rb +59 -18
- data/lib/nmatrix/monkeys.rb +0 -52
- data/lib/nmatrix/nmatrix.rb +136 -9
- data/lib/nmatrix/nvector.rb +33 -0
- data/lib/nmatrix/shortcuts.rb +95 -16
- data/lib/nmatrix/version.rb +1 -1
- data/lib/nmatrix/yale_functions.rb +25 -19
- data/spec/blas_spec.rb +1 -19
- data/spec/elementwise_spec.rb +132 -17
- data/spec/lapack_spec.rb +0 -3
- data/spec/nmatrix_list_spec.rb +18 -0
- data/spec/nmatrix_spec.rb +44 -18
- data/spec/nmatrix_yale_spec.rb +1 -3
- data/spec/shortcuts_spec.rb +26 -36
- data/spec/slice_spec.rb +2 -4
- metadata +2 -2
data/ext/nmatrix/data/rational.h
CHANGED
@@ -96,6 +96,7 @@ class RubyObject {
|
|
96
96
|
inline RubyObject(int64_t other) : rval(INT2FIX(other)) {}
|
97
97
|
// inline RubyObject(uint64_t other) : rval(INT2FIX(other)) {}
|
98
98
|
|
99
|
+
|
99
100
|
/*
|
100
101
|
* Float constructor.
|
101
102
|
*
|
@@ -103,7 +104,40 @@ class RubyObject {
|
|
103
104
|
*/
|
104
105
|
inline RubyObject(float other) : rval(rb_float_new(other)) {}
|
105
106
|
inline RubyObject(double other) : rval(rb_float_new(other)) {}
|
106
|
-
|
107
|
+
|
108
|
+
/*
|
109
|
+
* Operators for converting RubyObjects to other C types.
|
110
|
+
*/
|
111
|
+
|
112
|
+
#define RETURN_OBJ2NUM(mac) if (this->rval == Qtrue) return 1; else if (this->rval == Qfalse) return 0; else return mac(this->rval);
|
113
|
+
|
114
|
+
inline operator int8_t() const { RETURN_OBJ2NUM(NUM2INT) }
|
115
|
+
inline operator uint8_t() const { RETURN_OBJ2NUM(NUM2UINT) }
|
116
|
+
inline operator int16_t() const { RETURN_OBJ2NUM(NUM2INT) }
|
117
|
+
inline operator uint16_t() const { RETURN_OBJ2NUM(NUM2UINT) }
|
118
|
+
inline operator int32_t() const { RETURN_OBJ2NUM(NUM2LONG) }
|
119
|
+
inline operator VALUE() const { return rval; }
|
120
|
+
//inline operator uint32_t() const { return NUM2ULONG(this->rval); }
|
121
|
+
inline operator int64_t() const { RETURN_OBJ2NUM(NUM2LONG) }
|
122
|
+
inline operator uint64_t() const { RETURN_OBJ2NUM(NUM2ULONG) }
|
123
|
+
inline operator double() const { RETURN_OBJ2NUM(NUM2DBL) }
|
124
|
+
inline operator float() const { RETURN_OBJ2NUM(NUM2DBL) }
|
125
|
+
|
126
|
+
//template <typename IntType>
|
127
|
+
//inline operator Rational<typename std::enable_if<std::is_integral<IntType>::value, IntType>::type>() const { return this->to<Rational<IntType> >(); }
|
128
|
+
|
129
|
+
//template <typename IntType>
|
130
|
+
//inline operator Rational<typename std::enable_if<std::is_integral<IntType>::value, IntType>::type>&() const { static Rational<IntType> x = this->to<Rational<IntType> >(); return x; }
|
131
|
+
//inline operator Rational32() const { return this->to<Rational32>(); }
|
132
|
+
//inline operator Rational64() const { return this->to<Rational64>(); }
|
133
|
+
//inline operator Rational128() const { return this->to<Rational128>(); }
|
134
|
+
|
135
|
+
//template <typename FloatType>
|
136
|
+
//inline operator Complex<typename std::enable_if<std::is_floating_point<FloatType>::value, FloatType>::type>() const { return this->to<Complex<FloatType> >(); }
|
137
|
+
|
138
|
+
//template <typename FloatType>
|
139
|
+
//inline operator Complex<typename std::enable_if<std::is_floating_point<FloatType>::value, FloatType>::type>&() const { static Complex<FloatType> x = this->to<Complex<FloatType> >(); return x; }
|
140
|
+
|
107
141
|
/*
|
108
142
|
* Copy constructors.
|
109
143
|
*/
|
@@ -278,7 +312,7 @@ class RubyObject {
|
|
278
312
|
* Convert a Ruby object to a complex number.
|
279
313
|
*/
|
280
314
|
template <typename ComplexType>
|
281
|
-
inline typename std::enable_if<made_from_same_template<ComplexType, Complex64>::value, ComplexType>::type to(void) {
|
315
|
+
inline typename std::enable_if<made_from_same_template<ComplexType, Complex64>::value, ComplexType>::type to(void) const {
|
282
316
|
if (FIXNUM_P(this->rval) or TYPE(this->rval) == T_FLOAT or TYPE(this->rval) == T_RATIONAL) {
|
283
317
|
return ComplexType(NUM2DBL(this->rval));
|
284
318
|
|
@@ -294,7 +328,7 @@ class RubyObject {
|
|
294
328
|
* Convert a Ruby object to a rational number.
|
295
329
|
*/
|
296
330
|
template <typename RationalType>
|
297
|
-
inline typename std::enable_if<made_from_same_template<RationalType, Rational32>::value, RationalType>::type to(void) {
|
331
|
+
inline typename std::enable_if<made_from_same_template<RationalType, Rational32>::value, RationalType>::type to(void) const {
|
298
332
|
if (FIXNUM_P(this->rval) or TYPE(this->rval) == T_FLOAT or TYPE(this->rval) == T_COMPLEX) {
|
299
333
|
return RationalType(NUM2INT(this->rval));
|
300
334
|
|
@@ -305,11 +339,7 @@ class RubyObject {
|
|
305
339
|
rb_raise(rb_eTypeError, "Invalid conversion to Rational type.");
|
306
340
|
}
|
307
341
|
}
|
308
|
-
|
309
|
-
template <typename OtherType>
|
310
|
-
inline operator OtherType () {
|
311
|
-
return to<OtherType>();
|
312
|
-
}
|
342
|
+
|
313
343
|
};
|
314
344
|
|
315
345
|
// Negative operator
|
data/ext/nmatrix/extconf.rb
CHANGED
@@ -164,10 +164,14 @@ $objs = %w{nmatrix ruby_constants data/data util/io util/math util/sl_list stora
|
|
164
164
|
CONFIG['CXX'] = 'g++'
|
165
165
|
|
166
166
|
def find_newer_gplusplus #:nodoc:
|
167
|
-
|
168
|
-
|
167
|
+
print "checking for apparent GNU g++ binary with C++0x/C++11 support... "
|
168
|
+
[9,8,7,6,5,4,3].each do |minor|
|
169
|
+
ver = "4.#{minor}"
|
170
|
+
gpp = "g++-#{ver}"
|
171
|
+
result = `which #{gpp}`
|
169
172
|
next if result.empty?
|
170
|
-
CONFIG['CXX'] =
|
173
|
+
CONFIG['CXX'] = gpp
|
174
|
+
puts ver
|
171
175
|
return CONFIG['CXX']
|
172
176
|
end
|
173
177
|
false
|
@@ -195,13 +199,15 @@ else
|
|
195
199
|
else
|
196
200
|
$CPP_STANDARD = 'c++11'
|
197
201
|
end
|
202
|
+
puts "using C++ standard... #{$CPP_STANDARD}"
|
203
|
+
puts "g++ reports version... " + `#{CONFIG['CXX']} --version|head -n 1|cut -f 3 -d " "`
|
198
204
|
end
|
199
205
|
|
200
206
|
# For release, these next two should both be changed to -O3.
|
201
|
-
|
202
|
-
|
203
|
-
|
204
|
-
|
207
|
+
#$CFLAGS += " -O3 " #" -O0 -g "
|
208
|
+
$CFLAGS += " -static -O0 -g "
|
209
|
+
#$CPPFLAGS += " -O3 -std=#{$CPP_STANDARD} " #" -O0 -g -std=#{$CPP_STANDARD} " #-fmax-errors=10 -save-temps
|
210
|
+
$CPPFLAGS += " -static -O0 -g -std=#{$CPP_STANDARD} "
|
205
211
|
|
206
212
|
CONFIG['warnflags'].gsub!('-Wshorten-64-to-32', '') # doesn't work except in Mac-patched gcc (4.2)
|
207
213
|
CONFIG['warnflags'].gsub!('-Wdeclaration-after-statement', '')
|
data/ext/nmatrix/nmatrix.cpp
CHANGED
@@ -50,6 +50,8 @@ extern "C" {
|
|
50
50
|
#include "util/math.h"
|
51
51
|
#include "util/io.h"
|
52
52
|
#include "storage/storage.h"
|
53
|
+
#include "storage/list.h"
|
54
|
+
#include "storage/yale.h"
|
53
55
|
|
54
56
|
#include "nmatrix.h"
|
55
57
|
|
@@ -329,21 +331,19 @@ extern "C" {
|
|
329
331
|
static VALUE nm_init(int argc, VALUE* argv, VALUE nm);
|
330
332
|
static VALUE nm_init_copy(VALUE copy, VALUE original);
|
331
333
|
static VALUE nm_init_transposed(VALUE self);
|
332
|
-
static VALUE
|
334
|
+
static VALUE nm_cast(VALUE self, VALUE new_stype_symbol, VALUE new_dtype_symbol, VALUE init);
|
333
335
|
static VALUE nm_read(int argc, VALUE* argv, VALUE self);
|
334
336
|
static VALUE nm_write(int argc, VALUE* argv, VALUE self);
|
335
|
-
static VALUE nm_to_hash(VALUE self);
|
336
337
|
static VALUE nm_init_yale_from_old_yale(VALUE shape, VALUE dtype, VALUE ia, VALUE ja, VALUE a, VALUE from_dtype, VALUE nm);
|
337
338
|
static VALUE nm_alloc(VALUE klass);
|
338
|
-
static void nm_delete(NMATRIX* mat);
|
339
|
-
static void nm_delete_ref(NMATRIX* mat);
|
340
339
|
static VALUE nm_dtype(VALUE self);
|
341
340
|
static VALUE nm_itype(VALUE self);
|
342
341
|
static VALUE nm_stype(VALUE self);
|
342
|
+
static VALUE nm_default_value(VALUE self);
|
343
343
|
static VALUE nm_dim(VALUE self);
|
344
344
|
static VALUE nm_shape(VALUE self);
|
345
345
|
static VALUE nm_capacity(VALUE self);
|
346
|
-
static VALUE
|
346
|
+
static VALUE nm_each_with_indices(VALUE nmatrix);
|
347
347
|
static VALUE nm_each_stored_with_indices(VALUE nmatrix);
|
348
348
|
|
349
349
|
static SLICE* get_slice(size_t dim, VALUE* c, VALUE self);
|
@@ -355,6 +355,9 @@ static VALUE nm_is_ref(VALUE self);
|
|
355
355
|
|
356
356
|
static VALUE is_symmetric(VALUE self, bool hermitian);
|
357
357
|
|
358
|
+
static VALUE nm_guess_dtype(VALUE self, VALUE v);
|
359
|
+
static VALUE nm_min_dtype(VALUE self, VALUE v);
|
360
|
+
|
358
361
|
/*
|
359
362
|
* Macro defines an element-wise accessor function for some operation.
|
360
363
|
*
|
@@ -374,6 +377,8 @@ DECL_ELEMENTWISE_RUBY_ACCESSOR(add)
|
|
374
377
|
DECL_ELEMENTWISE_RUBY_ACCESSOR(subtract)
|
375
378
|
DECL_ELEMENTWISE_RUBY_ACCESSOR(multiply)
|
376
379
|
DECL_ELEMENTWISE_RUBY_ACCESSOR(divide)
|
380
|
+
DECL_ELEMENTWISE_RUBY_ACCESSOR(power)
|
381
|
+
DECL_ELEMENTWISE_RUBY_ACCESSOR(mod)
|
377
382
|
DECL_ELEMENTWISE_RUBY_ACCESSOR(eqeq)
|
378
383
|
DECL_ELEMENTWISE_RUBY_ACCESSOR(neq)
|
379
384
|
DECL_ELEMENTWISE_RUBY_ACCESSOR(lt)
|
@@ -417,17 +422,17 @@ void Init_nmatrix() {
|
|
417
422
|
///////////////////////
|
418
423
|
// Class Definitions //
|
419
424
|
///////////////////////
|
420
|
-
|
425
|
+
|
421
426
|
cNMatrix = rb_define_class("NMatrix", rb_cObject);
|
422
427
|
cNVector = rb_define_class("NVector", cNMatrix);
|
423
|
-
|
428
|
+
|
424
429
|
// Special exceptions
|
425
|
-
|
430
|
+
|
426
431
|
/*
|
427
432
|
* Exception raised when there's a problem with data.
|
428
433
|
*/
|
429
434
|
nm_eDataTypeError = rb_define_class("DataTypeError", rb_eStandardError);
|
430
|
-
|
435
|
+
|
431
436
|
/*
|
432
437
|
* Exception raised when something goes wrong with the storage of a matrix.
|
433
438
|
*/
|
@@ -436,7 +441,7 @@ void Init_nmatrix() {
|
|
436
441
|
///////////////////
|
437
442
|
// Class Methods //
|
438
443
|
///////////////////
|
439
|
-
|
444
|
+
|
440
445
|
rb_define_alloc_func(cNMatrix, nm_alloc);
|
441
446
|
|
442
447
|
///////////////////////
|
@@ -445,6 +450,8 @@ void Init_nmatrix() {
|
|
445
450
|
|
446
451
|
rb_define_singleton_method(cNMatrix, "upcast", (METHOD)nm_upcast, 2);
|
447
452
|
rb_define_singleton_method(cNMatrix, "itype_by_shape", (METHOD)nm_itype_by_shape, 1);
|
453
|
+
rb_define_singleton_method(cNMatrix, "guess_dtype", (METHOD)nm_guess_dtype, 1);
|
454
|
+
rb_define_singleton_method(cNMatrix, "min_dtype", (METHOD)nm_min_dtype, 1);
|
448
455
|
|
449
456
|
//////////////////////
|
450
457
|
// Instance Methods //
|
@@ -462,7 +469,10 @@ void Init_nmatrix() {
|
|
462
469
|
rb_define_method(cNMatrix, "dtype", (METHOD)nm_dtype, 0);
|
463
470
|
rb_define_method(cNMatrix, "itype", (METHOD)nm_itype, 0);
|
464
471
|
rb_define_method(cNMatrix, "stype", (METHOD)nm_stype, 0);
|
465
|
-
rb_define_method(cNMatrix, "
|
472
|
+
rb_define_method(cNMatrix, "cast_full", (METHOD)nm_cast, 3);
|
473
|
+
rb_define_method(cNMatrix, "default_value", (METHOD)nm_default_value, 0);
|
474
|
+
rb_define_protected_method(cNMatrix, "__list_default_value__", (METHOD)nm_list_default_value, 0);
|
475
|
+
rb_define_protected_method(cNMatrix, "__yale_default_value__", (METHOD)nm_yale_default_value, 0);
|
466
476
|
|
467
477
|
rb_define_method(cNMatrix, "[]", (METHOD)nm_mref, -1);
|
468
478
|
rb_define_method(cNMatrix, "slice", (METHOD)nm_mget, -1);
|
@@ -470,16 +480,21 @@ void Init_nmatrix() {
|
|
470
480
|
rb_define_method(cNMatrix, "is_ref?", (METHOD)nm_is_ref, 0);
|
471
481
|
rb_define_method(cNMatrix, "dimensions", (METHOD)nm_dim, 0);
|
472
482
|
|
473
|
-
rb_define_protected_method(cNMatrix, "
|
474
|
-
//rb_define_alias(cNMatrix, "to_h", "to_hash");
|
483
|
+
rb_define_protected_method(cNMatrix, "__list_to_hash__", (METHOD)nm_to_hash, 0); // handles list and dense, which are n-dimensional
|
475
484
|
|
476
485
|
rb_define_method(cNMatrix, "shape", (METHOD)nm_shape, 0);
|
477
486
|
rb_define_method(cNMatrix, "det_exact", (METHOD)nm_det_exact, 0);
|
478
487
|
//rb_define_method(cNMatrix, "transpose!", (METHOD)nm_transpose_self, 0);
|
479
488
|
rb_define_method(cNMatrix, "complex_conjugate!", (METHOD)nm_complex_conjugate_bang, 0);
|
480
489
|
|
481
|
-
|
490
|
+
rb_define_protected_method(cNMatrix, "__dense_each__", (METHOD)nm_dense_each, 0);
|
491
|
+
rb_define_protected_method(cNMatrix, "__dense_map__", (METHOD)nm_dense_map, 0);
|
492
|
+
rb_define_protected_method(cNMatrix, "__dense_map_pair__", (METHOD)nm_dense_map_pair, 1);
|
493
|
+
rb_define_method(cNMatrix, "each_with_indices", (METHOD)nm_each_with_indices, 0);
|
482
494
|
rb_define_method(cNMatrix, "each_stored_with_indices", (METHOD)nm_each_stored_with_indices, 0);
|
495
|
+
rb_define_protected_method(cNMatrix, "__list_map_merged_stored__", (METHOD)nm_list_map_merged_stored, 2);
|
496
|
+
rb_define_protected_method(cNMatrix, "__yale_map_merged_stored__", (METHOD)nm_yale_map_merged_stored, 2);
|
497
|
+
rb_define_protected_method(cNMatrix, "__yale_map_stored__", (METHOD)nm_yale_map_stored, 0);
|
483
498
|
|
484
499
|
rb_define_method(cNMatrix, "==", (METHOD)nm_eqeq, 1);
|
485
500
|
|
@@ -487,7 +502,8 @@ void Init_nmatrix() {
|
|
487
502
|
rb_define_method(cNMatrix, "-", (METHOD)nm_ew_subtract, 1);
|
488
503
|
rb_define_method(cNMatrix, "*", (METHOD)nm_ew_multiply, 1);
|
489
504
|
rb_define_method(cNMatrix, "/", (METHOD)nm_ew_divide, 1);
|
490
|
-
|
505
|
+
rb_define_method(cNMatrix, "**", (METHOD)nm_ew_power, 1);
|
506
|
+
rb_define_method(cNMatrix, "%", (METHOD)nm_ew_mod, 1);
|
491
507
|
|
492
508
|
rb_define_method(cNMatrix, "=~", (METHOD)nm_ew_eqeq, 1);
|
493
509
|
rb_define_method(cNMatrix, "!~", (METHOD)nm_ew_neq, 1);
|
@@ -496,6 +512,11 @@ void Init_nmatrix() {
|
|
496
512
|
rb_define_method(cNMatrix, "<", (METHOD)nm_ew_lt, 1);
|
497
513
|
rb_define_method(cNMatrix, ">", (METHOD)nm_ew_gt, 1);
|
498
514
|
|
515
|
+
/////////////////////////////
|
516
|
+
// Helper Instance Methods //
|
517
|
+
/////////////////////////////
|
518
|
+
rb_define_protected_method(cNMatrix, "__yale_vector_set__", (METHOD)nm_vector_set, -1);
|
519
|
+
|
499
520
|
/////////////////////////
|
500
521
|
// Matrix Math Methods //
|
501
522
|
/////////////////////////
|
@@ -507,18 +528,18 @@ void Init_nmatrix() {
|
|
507
528
|
rb_define_method(cNMatrix, "hermitian?", (METHOD)nm_hermitian, 0);
|
508
529
|
|
509
530
|
rb_define_method(cNMatrix, "capacity", (METHOD)nm_capacity, 0);
|
510
|
-
|
531
|
+
|
511
532
|
/////////////
|
512
533
|
// Aliases //
|
513
534
|
/////////////
|
514
|
-
|
535
|
+
|
515
536
|
rb_define_alias(cNMatrix, "dim", "dimensions");
|
516
537
|
rb_define_alias(cNMatrix, "equal?", "eql?");
|
517
|
-
|
538
|
+
|
518
539
|
///////////////////////
|
519
540
|
// Symbol Generation //
|
520
541
|
///////////////////////
|
521
|
-
|
542
|
+
|
522
543
|
nm_init_ruby_constants();
|
523
544
|
|
524
545
|
//////////////////////////
|
@@ -611,7 +632,7 @@ static VALUE nm_capacity(VALUE self) {
|
|
611
632
|
/*
|
612
633
|
* Destructor.
|
613
634
|
*/
|
614
|
-
|
635
|
+
void nm_delete(NMATRIX* mat) {
|
615
636
|
static void (*ttable[nm::NUM_STYPES])(STORAGE*) = {
|
616
637
|
nm_dense_storage_delete,
|
617
638
|
nm_list_storage_delete,
|
@@ -625,10 +646,10 @@ static void nm_delete(NMATRIX* mat) {
|
|
625
646
|
/*
|
626
647
|
* Slicing destructor.
|
627
648
|
*/
|
628
|
-
|
649
|
+
void nm_delete_ref(NMATRIX* mat) {
|
629
650
|
static void (*ttable[nm::NUM_STYPES])(STORAGE*) = {
|
630
651
|
nm_dense_storage_delete_ref,
|
631
|
-
nm_list_storage_delete_ref,
|
652
|
+
nm_list_storage_delete_ref,
|
632
653
|
nm_yale_storage_delete
|
633
654
|
};
|
634
655
|
ttable[mat->stype](mat->storage);
|
@@ -679,7 +700,7 @@ static VALUE nm_itype_by_shape(VALUE self, VALUE shape_arg) {
|
|
679
700
|
|
680
701
|
/*
|
681
702
|
* call-seq:
|
682
|
-
* upcast(first_dtype, second_dtype) ->
|
703
|
+
* upcast(first_dtype, second_dtype) -> Symbol
|
683
704
|
*
|
684
705
|
* Given a binary operation between types t1 and t2, what type will be returned?
|
685
706
|
*
|
@@ -694,28 +715,52 @@ static VALUE nm_upcast(VALUE self, VALUE t1, VALUE t2) {
|
|
694
715
|
}
|
695
716
|
|
696
717
|
|
697
|
-
|
698
718
|
/*
|
699
719
|
* call-seq:
|
700
|
-
|
720
|
+
default_value -> ...
|
701
721
|
*
|
702
|
-
*
|
722
|
+
* Get the default value for the matrix. For dense, this is undefined and will return Qnil. For list, it is user-defined.
|
723
|
+
* For yale, it's going to be some variation on zero, but may be Qfalse or Qnil.
|
724
|
+
*/
|
725
|
+
static VALUE nm_default_value(VALUE self) {
|
726
|
+
switch(NM_STYPE(self)) {
|
727
|
+
case nm::YALE_STORE:
|
728
|
+
return nm_yale_default_value(self);
|
729
|
+
case nm::LIST_STORE:
|
730
|
+
return nm_list_default_value(self);
|
731
|
+
case nm::DENSE_STORE:
|
732
|
+
default:
|
733
|
+
return Qnil;
|
734
|
+
}
|
735
|
+
}
|
736
|
+
|
737
|
+
|
738
|
+
/*
|
739
|
+
* call-seq:
|
740
|
+
* each_with_indices -> Enumerator
|
703
741
|
*
|
704
|
-
*
|
742
|
+
* Iterate over all entries of any matrix in standard storage order (as with #each), and include the indices.
|
705
743
|
*/
|
706
|
-
static VALUE
|
707
|
-
volatile VALUE nm = nmatrix;
|
744
|
+
static VALUE nm_each_with_indices(VALUE nmatrix) {
|
745
|
+
volatile VALUE nm = nmatrix;
|
708
746
|
|
709
747
|
switch(NM_STYPE(nm)) {
|
748
|
+
case nm::YALE_STORE:
|
749
|
+
return nm_yale_each_with_indices(nm);
|
710
750
|
case nm::DENSE_STORE:
|
711
|
-
return
|
751
|
+
return nm_dense_each_with_indices(nm);
|
752
|
+
case nm::LIST_STORE:
|
753
|
+
return nm_list_each_with_indices(nm, false);
|
712
754
|
default:
|
713
|
-
rb_raise(
|
755
|
+
rb_raise(nm_eDataTypeError, "Not a proper storage type");
|
714
756
|
}
|
715
757
|
}
|
716
758
|
|
717
759
|
/*
|
718
|
-
*
|
760
|
+
* call-seq:
|
761
|
+
* each_stored_with_indices -> Enumerator
|
762
|
+
*
|
763
|
+
* Iterate over the stored entries of any matrix. For dense and yale, this iterates over non-zero
|
719
764
|
* entries; for list, this iterates over non-default entries. Yields dim+1 values for each entry:
|
720
765
|
* i, j, ..., and the entry itself.
|
721
766
|
*/
|
@@ -728,14 +773,13 @@ static VALUE nm_each_stored_with_indices(VALUE nmatrix) {
|
|
728
773
|
case nm::DENSE_STORE:
|
729
774
|
return nm_dense_each_with_indices(nm);
|
730
775
|
case nm::LIST_STORE:
|
731
|
-
return
|
776
|
+
return nm_list_each_with_indices(nm, true);
|
732
777
|
default:
|
733
778
|
rb_raise(nm_eDataTypeError, "Not a proper storage type");
|
734
779
|
}
|
735
780
|
}
|
736
781
|
|
737
782
|
|
738
|
-
|
739
783
|
/*
|
740
784
|
* Equality operator. Returns a single true or false value indicating whether
|
741
785
|
* the matrices are equivalent.
|
@@ -777,7 +821,8 @@ DEF_ELEMENTWISE_RUBY_ACCESSOR(ADD, add)
|
|
777
821
|
DEF_ELEMENTWISE_RUBY_ACCESSOR(SUB, subtract)
|
778
822
|
DEF_ELEMENTWISE_RUBY_ACCESSOR(MUL, multiply)
|
779
823
|
DEF_ELEMENTWISE_RUBY_ACCESSOR(DIV, divide)
|
780
|
-
|
824
|
+
DEF_ELEMENTWISE_RUBY_ACCESSOR(POW, power)
|
825
|
+
DEF_ELEMENTWISE_RUBY_ACCESSOR(MOD, mod)
|
781
826
|
DEF_ELEMENTWISE_RUBY_ACCESSOR(EQEQ, eqeq)
|
782
827
|
DEF_ELEMENTWISE_RUBY_ACCESSOR(NEQ, neq)
|
783
828
|
DEF_ELEMENTWISE_RUBY_ACCESSOR(LEQ, leq)
|
@@ -927,7 +972,7 @@ static VALUE nm_init(int argc, VALUE* argv, VALUE nm) {
|
|
927
972
|
|
928
973
|
if (!SYMBOL_P(argv[0]) && TYPE(argv[0]) != T_STRING) {
|
929
974
|
stype = nm::DENSE_STORE;
|
930
|
-
|
975
|
+
|
931
976
|
} else {
|
932
977
|
// 0: String or Symbol
|
933
978
|
stype = interpret_stype(argv[0]);
|
@@ -938,12 +983,12 @@ static VALUE nm_init(int argc, VALUE* argv, VALUE nm) {
|
|
938
983
|
if (argc == 7) {
|
939
984
|
if (stype == nm::YALE_STORE) {
|
940
985
|
return nm_init_yale_from_old_yale(argv[1], argv[2], argv[3], argv[4], argv[5], argv[6], nm);
|
941
|
-
|
986
|
+
|
942
987
|
} else {
|
943
988
|
rb_raise(rb_eArgError, "Expected 2-4 arguments (or 7 for internal Yale creation)");
|
944
989
|
}
|
945
990
|
}
|
946
|
-
|
991
|
+
|
947
992
|
// 1: Array or Fixnum
|
948
993
|
size_t dim;
|
949
994
|
size_t* shape = interpret_shape(argv[offset], &dim);
|
@@ -953,23 +998,23 @@ static VALUE nm_init(int argc, VALUE* argv, VALUE nm) {
|
|
953
998
|
|
954
999
|
size_t init_cap = 0, init_val_len = 0;
|
955
1000
|
void* init_val = NULL;
|
956
|
-
if (
|
1001
|
+
if (!SYMBOL_P(argv[1+offset]) || TYPE(argv[1+offset]) == T_ARRAY) {
|
957
1002
|
// Initial value provided (could also be initial capacity, if yale).
|
958
|
-
|
959
|
-
if (stype == nm::YALE_STORE) {
|
1003
|
+
|
1004
|
+
if (stype == nm::YALE_STORE && NM_RUBYVAL_IS_NUMERIC(argv[1+offset])) {
|
960
1005
|
init_cap = FIX2UINT(argv[1+offset]);
|
961
|
-
|
1006
|
+
|
962
1007
|
} else {
|
963
1008
|
// 4: initial value / dtype
|
964
1009
|
init_val = interpret_initial_value(argv[1+offset], dtype);
|
965
|
-
|
1010
|
+
|
966
1011
|
if (TYPE(argv[1+offset]) == T_ARRAY) init_val_len = RARRAY_LEN(argv[1+offset]);
|
967
1012
|
else init_val_len = 1;
|
968
1013
|
}
|
969
|
-
|
1014
|
+
|
970
1015
|
} else {
|
971
1016
|
// DType is RUBYOBJ.
|
972
|
-
|
1017
|
+
|
973
1018
|
if (stype == nm::DENSE_STORE) {
|
974
1019
|
/*
|
975
1020
|
* No need to initialize dense with any kind of default value unless it's
|
@@ -979,9 +1024,9 @@ static VALUE nm_init(int argc, VALUE* argv, VALUE nm) {
|
|
979
1024
|
// Pretend [nil] was passed for RUBYOBJ.
|
980
1025
|
init_val = ALLOC(VALUE);
|
981
1026
|
*(VALUE*)init_val = Qnil;
|
982
|
-
|
1027
|
+
|
983
1028
|
init_val_len = 1;
|
984
|
-
|
1029
|
+
|
985
1030
|
} else {
|
986
1031
|
init_val = NULL;
|
987
1032
|
}
|
@@ -990,51 +1035,40 @@ static VALUE nm_init(int argc, VALUE* argv, VALUE nm) {
|
|
990
1035
|
std::memset(init_val, 0, DTYPE_SIZES[dtype]);
|
991
1036
|
}
|
992
1037
|
}
|
993
|
-
|
1038
|
+
|
994
1039
|
// TODO: Update to allow an array as the initial value.
|
995
1040
|
NMATRIX* nmatrix;
|
996
1041
|
UnwrapNMatrix(nm, nmatrix);
|
997
1042
|
|
998
1043
|
nmatrix->stype = stype;
|
999
|
-
|
1044
|
+
|
1000
1045
|
switch (stype) {
|
1001
1046
|
case nm::DENSE_STORE:
|
1002
1047
|
nmatrix->storage = (STORAGE*)nm_dense_storage_create(dtype, shape, dim, init_val, init_val_len);
|
1003
1048
|
break;
|
1004
|
-
|
1049
|
+
|
1005
1050
|
case nm::LIST_STORE:
|
1006
1051
|
nmatrix->storage = (STORAGE*)nm_list_storage_create(dtype, shape, dim, init_val);
|
1007
1052
|
break;
|
1008
|
-
|
1053
|
+
|
1009
1054
|
case nm::YALE_STORE:
|
1010
1055
|
nmatrix->storage = (STORAGE*)nm_yale_storage_create(dtype, shape, dim, init_cap, nm::UINT8);
|
1011
|
-
nm_yale_storage_init((YALE_STORAGE*)(nmatrix->storage));
|
1056
|
+
nm_yale_storage_init((YALE_STORAGE*)(nmatrix->storage), NULL);
|
1012
1057
|
break;
|
1013
1058
|
}
|
1014
1059
|
|
1015
1060
|
return nm;
|
1016
1061
|
}
|
1017
1062
|
|
1063
|
+
|
1018
1064
|
/*
|
1019
1065
|
* call-seq:
|
1020
|
-
*
|
1021
|
-
*
|
1022
|
-
* Create a Ruby Hash from an NMatrix.
|
1066
|
+
* cast(stype) -> NMatrix
|
1067
|
+
* cast(stype, dtype, sparse_basis) -> NMatrix
|
1023
1068
|
*
|
1024
|
-
* This is an internal C function which handles list stype only.
|
1025
|
-
*/
|
1026
|
-
static VALUE nm_to_hash(VALUE self) {
|
1027
|
-
if (NM_STYPE(self) != nm::LIST_STORE) {
|
1028
|
-
rb_raise(rb_eNotImpError, "please cast to :list first");
|
1029
|
-
}
|
1030
|
-
|
1031
|
-
return nm_list_storage_to_hash(NM_STORAGE_LIST(self), NM_DTYPE(self));
|
1032
|
-
}
|
1033
|
-
|
1034
|
-
/*
|
1035
1069
|
* Copy constructor for changing dtypes and stypes.
|
1036
1070
|
*/
|
1037
|
-
static VALUE
|
1071
|
+
static VALUE nm_cast(VALUE self, VALUE new_stype_symbol, VALUE new_dtype_symbol, VALUE init) {
|
1038
1072
|
nm::dtype_t new_dtype = nm_dtype_from_rbsymbol(new_dtype_symbol);
|
1039
1073
|
nm::stype_t new_stype = nm_stype_from_rbsymbol(new_stype_symbol);
|
1040
1074
|
|
@@ -1046,9 +1080,12 @@ static VALUE nm_init_cast_copy(VALUE self, VALUE new_stype_symbol, VALUE new_dty
|
|
1046
1080
|
|
1047
1081
|
UnwrapNMatrix( self, rhs );
|
1048
1082
|
|
1083
|
+
void* init_ptr = ALLOCA_N(char, DTYPE_SIZES[new_dtype]);
|
1084
|
+
rubyval_to_cval(init, new_dtype, init_ptr);
|
1085
|
+
|
1049
1086
|
// Copy the storage
|
1050
|
-
|
1051
|
-
lhs->storage = cast_copy[lhs->stype][rhs->stype](rhs->storage, new_dtype);
|
1087
|
+
CAST_TABLE(cast_copy);
|
1088
|
+
lhs->storage = cast_copy[lhs->stype][rhs->stype](rhs->storage, new_dtype, init_ptr);
|
1052
1089
|
|
1053
1090
|
STYPE_MARK_TABLE(mark);
|
1054
1091
|
|
@@ -1090,8 +1127,8 @@ static VALUE nm_init_copy(VALUE copy, VALUE original) {
|
|
1090
1127
|
lhs->stype = rhs->stype;
|
1091
1128
|
|
1092
1129
|
// Copy the storage
|
1093
|
-
|
1094
|
-
lhs->storage = ttable[lhs->stype][rhs->stype](rhs->storage, rhs->storage->dtype);
|
1130
|
+
CAST_TABLE(ttable);
|
1131
|
+
lhs->storage = ttable[lhs->stype][rhs->stype](rhs->storage, rhs->storage->dtype, NULL);
|
1095
1132
|
|
1096
1133
|
return copy;
|
1097
1134
|
}
|
@@ -1299,7 +1336,7 @@ static VALUE nm_read(int argc, VALUE* argv, VALUE self) {
|
|
1299
1336
|
|
1300
1337
|
|
1301
1338
|
if (!RB_FILE_EXISTS(file)) { // FIXME: Errno::ENOENT
|
1302
|
-
rb_raise(rb_get_errno_exc("ENOENT"), RSTRING_PTR(file));
|
1339
|
+
rb_raise(rb_get_errno_exc("ENOENT"), "%s", RSTRING_PTR(file));
|
1303
1340
|
}
|
1304
1341
|
|
1305
1342
|
// Open a file stream
|
@@ -1373,7 +1410,7 @@ static VALUE nm_read(int argc, VALUE* argv, VALUE self) {
|
|
1373
1410
|
case nm::DENSE_STORE:
|
1374
1411
|
return Data_Wrap_Struct(klass, nm_dense_storage_mark, nm_delete, nm);
|
1375
1412
|
case nm::YALE_STORE:
|
1376
|
-
return Data_Wrap_Struct(
|
1413
|
+
return Data_Wrap_Struct(klass, nm_yale_storage_mark, nm_delete, nm);
|
1377
1414
|
default:
|
1378
1415
|
return Qnil;
|
1379
1416
|
}
|
@@ -1438,7 +1475,7 @@ static VALUE nm_mget(int argc, VALUE* argv, VALUE self) {
|
|
1438
1475
|
nm_list_storage_get,
|
1439
1476
|
nm_yale_storage_get
|
1440
1477
|
};
|
1441
|
-
|
1478
|
+
|
1442
1479
|
return nm_xslice(argc, argv, ttable[NM_STYPE(self)], nm_delete, self);
|
1443
1480
|
}
|
1444
1481
|
|
@@ -1551,7 +1588,7 @@ static VALUE nm_multiply(VALUE left_v, VALUE right_v) {
|
|
1551
1588
|
|
1552
1589
|
return matrix_multiply(left, right);
|
1553
1590
|
|
1554
|
-
}
|
1591
|
+
}
|
1555
1592
|
|
1556
1593
|
return Qnil;
|
1557
1594
|
}
|
@@ -1708,26 +1745,29 @@ static VALUE nm_xslice(int argc, VALUE* argv, void* (*slice_func)(STORAGE*, SLIC
|
|
1708
1745
|
static VALUE elementwise_op(nm::ewop_t op, VALUE left_val, VALUE right_val) {
|
1709
1746
|
STYPE_MARK_TABLE(mark);
|
1710
1747
|
|
1711
|
-
|
1712
|
-
|
1713
|
-
|
1714
|
-
nm_yale_storage_ew_op
|
1715
|
-
};
|
1716
|
-
|
1717
|
-
NMATRIX *result = ALLOC(NMATRIX), *left;
|
1718
|
-
|
1748
|
+
NMATRIX* left;
|
1749
|
+
NMATRIX* result;
|
1750
|
+
|
1719
1751
|
CheckNMatrixType(left_val);
|
1720
1752
|
UnwrapNMatrix(left_val, left);
|
1721
1753
|
|
1722
1754
|
if (TYPE(right_val) != T_DATA || (RDATA(right_val)->dfree != (RUBY_DATA_FUNC)nm_delete && RDATA(right_val)->dfree != (RUBY_DATA_FUNC)nm_delete_ref)) {
|
1723
1755
|
// This is a matrix-scalar element-wise operation.
|
1724
|
-
|
1725
|
-
|
1726
|
-
|
1727
|
-
|
1728
|
-
|
1729
|
-
|
1756
|
+
std::string sym;
|
1757
|
+
switch(left->stype) {
|
1758
|
+
case nm::DENSE_STORE:
|
1759
|
+
sym = "__dense_scalar_" + nm::EWOP_NAMES[op] + "__";
|
1760
|
+
break;
|
1761
|
+
case nm::YALE_STORE:
|
1762
|
+
sym = "__yale_scalar_" + nm::EWOP_NAMES[op] + "__";
|
1763
|
+
break;
|
1764
|
+
case nm::LIST_STORE:
|
1765
|
+
sym = "__list_scalar_" + nm::EWOP_NAMES[op] + "__";
|
1766
|
+
break;
|
1767
|
+
default:
|
1768
|
+
rb_raise(rb_eNotImpError, "unknown storage type requested scalar element-wise operation");
|
1730
1769
|
}
|
1770
|
+
return rb_funcall(left_val, rb_intern(sym.c_str()), 1, right_val);
|
1731
1771
|
|
1732
1772
|
} else {
|
1733
1773
|
|
@@ -1743,20 +1783,31 @@ static VALUE elementwise_op(nm::ewop_t op, VALUE left_val, VALUE right_val) {
|
|
1743
1783
|
|
1744
1784
|
NMATRIX* right;
|
1745
1785
|
UnwrapNMatrix(right_val, right);
|
1746
|
-
|
1747
|
-
if (left->stype == right->stype) {
|
1748
1786
|
|
1749
|
-
|
1750
|
-
|
1787
|
+
if (left->stype == right->stype) {
|
1788
|
+
std::string sym;
|
1789
|
+
|
1790
|
+
switch(left->stype) {
|
1791
|
+
case nm::DENSE_STORE:
|
1792
|
+
sym = "__dense_elementwise_" + nm::EWOP_NAMES[op] + "__";
|
1793
|
+
break;
|
1794
|
+
case nm::YALE_STORE:
|
1795
|
+
sym = "__yale_elementwise_" + nm::EWOP_NAMES[op] + "__";
|
1796
|
+
break;
|
1797
|
+
case nm::LIST_STORE:
|
1798
|
+
sym = "__list_elementwise_" + nm::EWOP_NAMES[op] + "__";
|
1799
|
+
break;
|
1800
|
+
default:
|
1801
|
+
rb_raise(rb_eNotImpError, "unknown storage type requested element-wise operation");
|
1802
|
+
}
|
1803
|
+
return rb_funcall(left_val, rb_intern(sym.c_str()), 1, right_val);
|
1751
1804
|
|
1752
1805
|
} else {
|
1753
1806
|
rb_raise(rb_eArgError, "Element-wise operations are not currently supported between matrices with differing stypes.");
|
1754
1807
|
}
|
1755
1808
|
}
|
1756
1809
|
|
1757
|
-
|
1758
|
-
|
1759
|
-
return result_val;
|
1810
|
+
return Data_Wrap_Struct(CLASS_OF(left_val), mark[result->stype], nm_delete, result);
|
1760
1811
|
}
|
1761
1812
|
|
1762
1813
|
/*
|
@@ -1767,7 +1818,7 @@ bool is_ref(const NMATRIX* matrix) {
|
|
1767
1818
|
if (matrix->stype != nm::DENSE_STORE) {
|
1768
1819
|
return false;
|
1769
1820
|
}
|
1770
|
-
|
1821
|
+
|
1771
1822
|
return ((DENSE_STORAGE*)(matrix->storage))->src != matrix->storage;
|
1772
1823
|
}
|
1773
1824
|
|
@@ -1782,11 +1833,11 @@ static VALUE is_symmetric(VALUE self, bool hermitian) {
|
|
1782
1833
|
if (NM_STYPE(self) == nm::DENSE_STORE) {
|
1783
1834
|
if (hermitian) {
|
1784
1835
|
nm_dense_storage_is_hermitian((DENSE_STORAGE*)(m->storage), m->storage->shape[0]);
|
1785
|
-
|
1836
|
+
|
1786
1837
|
} else {
|
1787
1838
|
nm_dense_storage_is_symmetric((DENSE_STORAGE*)(m->storage), m->storage->shape[0]);
|
1788
1839
|
}
|
1789
|
-
|
1840
|
+
|
1790
1841
|
} else {
|
1791
1842
|
// TODO: Implement, at the very least, yale_is_symmetric. Model it after yale/transp.template.c.
|
1792
1843
|
rb_raise(rb_eNotImpError, "symmetric? and hermitian? only implemented for dense currently");
|
@@ -1801,6 +1852,84 @@ static VALUE is_symmetric(VALUE self, bool hermitian) {
|
|
1801
1852
|
// Utility Functions //
|
1802
1853
|
///////////////////////
|
1803
1854
|
|
1855
|
+
/*
|
1856
|
+
* Guess the dtype given a Ruby VALUE and return it as a symbol.
|
1857
|
+
*
|
1858
|
+
* Not to be confused with nm_dtype_guess, which returns an nm::dtype_t. (This calls that.)
|
1859
|
+
*/
|
1860
|
+
static VALUE nm_guess_dtype(VALUE self, VALUE v) {
|
1861
|
+
return ID2SYM(rb_intern(DTYPE_NAMES[nm_dtype_guess(v)]));
|
1862
|
+
}
|
1863
|
+
|
1864
|
+
/*
|
1865
|
+
* Get the minimum allowable dtype for a Ruby VALUE and return it as a symbol.
|
1866
|
+
*/
|
1867
|
+
static VALUE nm_min_dtype(VALUE self, VALUE v) {
|
1868
|
+
return ID2SYM(rb_intern(DTYPE_NAMES[nm_dtype_min(v)]));
|
1869
|
+
}
|
1870
|
+
|
1871
|
+
/*
|
1872
|
+
* Helper for nm_dtype_min(), handling integers.
|
1873
|
+
*/
|
1874
|
+
nm::dtype_t nm_dtype_min_fixnum(int64_t v) {
|
1875
|
+
if (v >= 0 && v <= UCHAR_MAX) return nm::BYTE;
|
1876
|
+
else {
|
1877
|
+
v = std::abs(v);
|
1878
|
+
if (v <= CHAR_MAX) return nm::INT8;
|
1879
|
+
else if (v <= SHRT_MAX) return nm::INT16;
|
1880
|
+
else if (v <= INT_MAX) return nm::INT32;
|
1881
|
+
else return nm::INT64;
|
1882
|
+
}
|
1883
|
+
}
|
1884
|
+
|
1885
|
+
/*
|
1886
|
+
* Helper for nm_dtype_min(), handling rationals.
|
1887
|
+
*/
|
1888
|
+
nm::dtype_t nm_dtype_min_rational(VALUE vv) {
|
1889
|
+
nm::Rational128* v = ALLOCA_N(nm::Rational128, 1);
|
1890
|
+
rubyval_to_cval(vv, nm::RATIONAL128, v);
|
1891
|
+
|
1892
|
+
int64_t i = std::max(std::abs(v->n), v->d);
|
1893
|
+
if (i <= SHRT_MAX) return nm::INT16;
|
1894
|
+
else if (i <= INT_MAX) return nm::INT32;
|
1895
|
+
else return nm::INT64;
|
1896
|
+
}
|
1897
|
+
|
1898
|
+
/*
|
1899
|
+
* Return the minimum dtype required to store a given value.
|
1900
|
+
*
|
1901
|
+
* This is kind of arbitrary. For Float, it always returns :float32 for example, since in some cases neither :float64
|
1902
|
+
* not :float32 are sufficient.
|
1903
|
+
*
|
1904
|
+
* This function is used in upcasting for scalar math. We want to ensure that :int8 + 1 does not return an :int64, basically.
|
1905
|
+
*
|
1906
|
+
* FIXME: Eventually, this function should actually look at the value stored in Fixnums (for example), so that it knows
|
1907
|
+
* whether to return :int64 or :int32.
|
1908
|
+
*/
|
1909
|
+
nm::dtype_t nm_dtype_min(VALUE v) {
|
1910
|
+
|
1911
|
+
switch(TYPE(v)) {
|
1912
|
+
case T_FIXNUM:
|
1913
|
+
return nm_dtype_min_fixnum(FIX2LONG(v));
|
1914
|
+
case T_BIGNUM:
|
1915
|
+
return nm::INT64;
|
1916
|
+
case T_FLOAT:
|
1917
|
+
return nm::FLOAT32;
|
1918
|
+
case T_COMPLEX:
|
1919
|
+
return nm::COMPLEX64;
|
1920
|
+
case T_RATIONAL:
|
1921
|
+
return nm_dtype_min_rational(v);
|
1922
|
+
case T_STRING:
|
1923
|
+
return RSTRING_LEN(v) == 1 ? nm::BYTE : nm::RUBYOBJ;
|
1924
|
+
case T_TRUE:
|
1925
|
+
case T_FALSE:
|
1926
|
+
case T_NIL:
|
1927
|
+
default:
|
1928
|
+
return nm::RUBYOBJ;
|
1929
|
+
}
|
1930
|
+
}
|
1931
|
+
|
1932
|
+
|
1804
1933
|
/*
|
1805
1934
|
* Guess the data type given a value.
|
1806
1935
|
*
|
@@ -1810,35 +1939,30 @@ nm::dtype_t nm_dtype_guess(VALUE v) {
|
|
1810
1939
|
switch(TYPE(v)) {
|
1811
1940
|
case T_TRUE:
|
1812
1941
|
case T_FALSE:
|
1813
|
-
|
1814
|
-
|
1942
|
+
case T_NIL:
|
1943
|
+
return nm::RUBYOBJ;
|
1815
1944
|
case T_STRING:
|
1816
|
-
|
1817
|
-
return nm::BYTE;
|
1818
|
-
|
1819
|
-
} else {
|
1820
|
-
rb_raise(rb_eArgError, "Strings of length > 1 may not be stored in a matrix.");
|
1821
|
-
}
|
1945
|
+
return RSTRING_LEN(v) == 1 ? nm::BYTE : nm::RUBYOBJ;
|
1822
1946
|
|
1823
1947
|
#if SIZEOF_INT == 8
|
1824
1948
|
case T_FIXNUM:
|
1825
1949
|
return nm::INT64;
|
1826
|
-
|
1950
|
+
|
1827
1951
|
case T_RATIONAL:
|
1828
1952
|
return nm::RATIONAL128;
|
1829
|
-
|
1953
|
+
|
1830
1954
|
#else
|
1831
1955
|
# if SIZEOF_INT == 4
|
1832
1956
|
case T_FIXNUM:
|
1833
1957
|
return nm::INT32;
|
1834
|
-
|
1958
|
+
|
1835
1959
|
case T_RATIONAL:
|
1836
1960
|
return nm::RATIONAL64;
|
1837
|
-
|
1961
|
+
|
1838
1962
|
#else
|
1839
1963
|
case T_FIXNUM:
|
1840
1964
|
return nm::INT16;
|
1841
|
-
|
1965
|
+
|
1842
1966
|
case T_RATIONAL:
|
1843
1967
|
return nm::RATIONAL32;
|
1844
1968
|
# endif
|
@@ -1850,15 +1974,15 @@ nm::dtype_t nm_dtype_guess(VALUE v) {
|
|
1850
1974
|
#if SIZEOF_FLOAT == 4
|
1851
1975
|
case T_COMPLEX:
|
1852
1976
|
return nm::COMPLEX128;
|
1853
|
-
|
1977
|
+
|
1854
1978
|
case T_FLOAT:
|
1855
1979
|
return nm::FLOAT64;
|
1856
|
-
|
1980
|
+
|
1857
1981
|
#else
|
1858
1982
|
# if SIZEOF_FLOAT == 2
|
1859
1983
|
case T_COMPLEX:
|
1860
1984
|
return nm::COMPLEX64;
|
1861
|
-
|
1985
|
+
|
1862
1986
|
case T_FLOAT:
|
1863
1987
|
return nm::FLOAT32;
|
1864
1988
|
# endif
|
@@ -1867,13 +1991,12 @@ nm::dtype_t nm_dtype_guess(VALUE v) {
|
|
1867
1991
|
case T_ARRAY:
|
1868
1992
|
/*
|
1869
1993
|
* May be passed for dense -- for now, just look at the first element.
|
1870
|
-
|
1994
|
+
*
|
1871
1995
|
* TODO: Look at entire array for most specific type.
|
1872
1996
|
*/
|
1873
|
-
|
1997
|
+
|
1874
1998
|
return nm_dtype_guess(RARRAY_PTR(v)[0]);
|
1875
1999
|
|
1876
|
-
case T_NIL:
|
1877
2000
|
default:
|
1878
2001
|
rb_raise(rb_eArgError, "Unable to guess a data type from provided parameters; data type must be specified manually.");
|
1879
2002
|
}
|
@@ -1928,9 +2051,9 @@ static SLICE* get_slice(size_t dim, VALUE* c, VALUE self) {
|
|
1928
2051
|
static double get_time(void) {
|
1929
2052
|
struct timeval t;
|
1930
2053
|
struct timezone tzp;
|
1931
|
-
|
2054
|
+
|
1932
2055
|
gettimeofday(&t, &tzp);
|
1933
|
-
|
2056
|
+
|
1934
2057
|
return t.tv_sec + t.tv_usec*1e-6;
|
1935
2058
|
}
|
1936
2059
|
#endif
|
@@ -1942,16 +2065,16 @@ static double get_time(void) {
|
|
1942
2065
|
*/
|
1943
2066
|
static nm::dtype_t interpret_dtype(int argc, VALUE* argv, nm::stype_t stype) {
|
1944
2067
|
int offset;
|
1945
|
-
|
2068
|
+
|
1946
2069
|
switch (argc) {
|
1947
2070
|
case 1:
|
1948
2071
|
offset = 0;
|
1949
2072
|
break;
|
1950
|
-
|
2073
|
+
|
1951
2074
|
case 2:
|
1952
2075
|
offset = 1;
|
1953
2076
|
break;
|
1954
|
-
|
2077
|
+
|
1955
2078
|
default:
|
1956
2079
|
rb_raise(rb_eArgError, "Need an initial value or a dtype.");
|
1957
2080
|
break;
|
@@ -1959,13 +2082,13 @@ static nm::dtype_t interpret_dtype(int argc, VALUE* argv, nm::stype_t stype) {
|
|
1959
2082
|
|
1960
2083
|
if (SYMBOL_P(argv[offset])) {
|
1961
2084
|
return nm_dtype_from_rbsymbol(argv[offset]);
|
1962
|
-
|
2085
|
+
|
1963
2086
|
} else if (TYPE(argv[offset]) == T_STRING) {
|
1964
2087
|
return nm_dtype_from_rbstring(StringValue(argv[offset]));
|
1965
|
-
|
2088
|
+
|
1966
2089
|
} else if (stype == nm::YALE_STORE) {
|
1967
2090
|
rb_raise(rb_eArgError, "Yale storage class requires a dtype.");
|
1968
|
-
|
2091
|
+
|
1969
2092
|
} else {
|
1970
2093
|
return nm_dtype_guess(argv[0]);
|
1971
2094
|
}
|
@@ -1977,18 +2100,18 @@ static nm::dtype_t interpret_dtype(int argc, VALUE* argv, nm::stype_t stype) {
|
|
1977
2100
|
static void* interpret_initial_value(VALUE arg, nm::dtype_t dtype) {
|
1978
2101
|
unsigned int index;
|
1979
2102
|
void* init_val;
|
1980
|
-
|
2103
|
+
|
1981
2104
|
if (TYPE(arg) == T_ARRAY) {
|
1982
2105
|
// Array
|
1983
|
-
|
2106
|
+
|
1984
2107
|
init_val = ALLOC_N(int8_t, DTYPE_SIZES[dtype] * RARRAY_LEN(arg));
|
1985
2108
|
for (index = 0; index < RARRAY_LEN(arg); ++index) {
|
1986
2109
|
rubyval_to_cval(RARRAY_PTR(arg)[index], dtype, (char*)init_val + (index * DTYPE_SIZES[dtype]));
|
1987
2110
|
}
|
1988
|
-
|
2111
|
+
|
1989
2112
|
} else {
|
1990
2113
|
// Single value
|
1991
|
-
|
2114
|
+
|
1992
2115
|
init_val = rubyobj_to_cval(arg, dtype);
|
1993
2116
|
}
|
1994
2117
|
|
@@ -2007,18 +2130,18 @@ static size_t* interpret_shape(VALUE arg, size_t* dim) {
|
|
2007
2130
|
if (TYPE(arg) == T_ARRAY) {
|
2008
2131
|
*dim = RARRAY_LEN(arg);
|
2009
2132
|
shape = ALLOC_N(size_t, *dim);
|
2010
|
-
|
2133
|
+
|
2011
2134
|
for (size_t index = 0; index < *dim; ++index) {
|
2012
2135
|
shape[index] = FIX2UINT( RARRAY_PTR(arg)[index] );
|
2013
2136
|
}
|
2014
|
-
|
2137
|
+
|
2015
2138
|
} else if (FIXNUM_P(arg)) {
|
2016
2139
|
*dim = 2;
|
2017
2140
|
shape = ALLOC_N(size_t, *dim);
|
2018
|
-
|
2141
|
+
|
2019
2142
|
shape[0] = FIX2UINT(arg);
|
2020
2143
|
shape[1] = FIX2UINT(arg);
|
2021
|
-
|
2144
|
+
|
2022
2145
|
} else {
|
2023
2146
|
rb_raise(rb_eArgError, "Expected an array of numbers or a single Fixnum for matrix shape");
|
2024
2147
|
}
|
@@ -2032,10 +2155,10 @@ static size_t* interpret_shape(VALUE arg, size_t* dim) {
|
|
2032
2155
|
static nm::stype_t interpret_stype(VALUE arg) {
|
2033
2156
|
if (SYMBOL_P(arg)) {
|
2034
2157
|
return nm_stype_from_rbsymbol(arg);
|
2035
|
-
|
2158
|
+
|
2036
2159
|
} else if (TYPE(arg) == T_STRING) {
|
2037
2160
|
return nm_stype_from_rbstring(StringValue(arg));
|
2038
|
-
|
2161
|
+
|
2039
2162
|
} else {
|
2040
2163
|
rb_raise(rb_eArgError, "Expected storage type");
|
2041
2164
|
}
|
@@ -2049,8 +2172,8 @@ STORAGE* matrix_storage_cast_alloc(NMATRIX* matrix, nm::dtype_t new_dtype) {
|
|
2049
2172
|
if (matrix->storage->dtype == new_dtype && !is_ref(matrix))
|
2050
2173
|
return matrix->storage;
|
2051
2174
|
|
2052
|
-
|
2053
|
-
return cast_copy_storage[matrix->stype][matrix->stype](matrix->storage, new_dtype);
|
2175
|
+
CAST_TABLE(cast_copy_storage);
|
2176
|
+
return cast_copy_storage[matrix->stype][matrix->stype](matrix->storage, new_dtype, NULL);
|
2054
2177
|
}
|
2055
2178
|
|
2056
2179
|
STORAGE_PAIR binary_storage_cast_alloc(NMATRIX* left_matrix, NMATRIX* right_matrix) {
|
@@ -2156,7 +2279,7 @@ VALUE rb_nmatrix_dense_create(nm::dtype_t dtype, size_t* shape, size_t dim, void
|
|
2156
2279
|
shape_copy = ALLOC_N(size_t, nm_dim);
|
2157
2280
|
shape_copy[0] = shape[0];
|
2158
2281
|
shape_copy[1] = 1;
|
2159
|
-
|
2282
|
+
|
2160
2283
|
} else {
|
2161
2284
|
klass = cNMatrix;
|
2162
2285
|
nm_dim = dim;
|