-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathvec.go
206 lines (189 loc) · 4.26 KB
/
vec.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
package lap
import (
"fmt"
"reflect"
"unsafe"
)
type Vector interface {
Matrix
AtVec(i int) float64
Len() int
}
type DenseV struct {
data []float64
incMinusOne int
}
// NewDenseVector returns a vector of length n with data. If data is nil it is
// automatically allocated.
func NewDenseVector(n int, data []float64) *DenseV {
if data == nil {
data = make([]float64, n)
}
if len(data) != n {
panic(ErrDim)
}
return &DenseV{
data: data,
}
}
// Set implements the matrixSetter interface.
func (v *DenseV) Set(i, j int, f float64) {
var _ matrixSetter = v
if j != 0 {
panic(ErrColAccess)
}
v.SetVec(i, f)
}
func (v *DenseV) Dims() (int, int) { return v.Len(), 1 }
func (v *DenseV) At(i, j int) float64 {
if j != 0 {
panic(ErrColAccess)
}
return v.AtVec(i)
}
func (v *DenseV) Len() int {
l := len(v.data)
if v.incMinusOne != 0 {
div, mod := l/(v.incMinusOne+1), l%(v.incMinusOne+1)
if mod == 0 {
return div
} else {
return div + 1
}
}
return l
}
func (v *DenseV) AtVec(i int) float64 {
if v.incMinusOne != 0 {
return v.data[i*(v.incMinusOne+1)]
}
return v.data[i]
}
func (v *DenseV) SetVec(i int, f float64) {
if v.incMinusOne != 0 {
v.data[i*(v.incMinusOne+1)] = f
} else {
v.data[i] = f
}
}
// AddVec adds the vectors a+b element-wise, placing the result in the receiver.
func (v *DenseV) AddVec(a, b Vector) {
n := v.Len()
if v.data == nil {
*v = *NewDenseVector(n, nil)
}
if n != b.Len() || n != a.Len() {
panic(ErrDim)
}
for i := 0; i < n; i++ {
v.SetVec(i, a.AtVec(i)+b.AtVec(i))
}
}
// SubVec subtracts the vectors a-b element-wise, placing the result in the receiver.
func (v *DenseV) SubVec(a, b Vector) {
n := v.Len()
if v.data == nil {
*v = *NewDenseVector(n, nil)
}
if n != b.Len() || n != a.Len() {
panic(ErrDim)
}
for i := 0; i < n; i++ {
v.SetVec(i, a.AtVec(i)-b.AtVec(i))
}
}
// CopyVec makes a copy of elements of a into the receiver and returns the amount
// of elements copied. If the receiver has not been initialized then a vector is allocated.
func (v *DenseV) CopyVec(a Vector) int {
n := a.Len()
if v.data == nil {
*v = *NewDenseVector(n, nil)
}
if n != v.Len() {
panic(ErrDim)
}
for i := 0; i < n; i++ {
v.SetVec(i, a.AtVec(i))
}
return n
}
// MulVec computes A * b. The result is stored into the receiver.
// MulVec panics if the number of columns in A does not equal the number of
// rows in b or if the number of columns in b does not equal 1.
func (v *DenseV) MulVec(A Matrix, b Vector) {
n := b.Len()
m, c := A.Dims()
if c != n {
panic(ErrDim)
}
if v.data == nil {
*v = *NewDenseVector(m, nil)
} else if aliasedData(v, b) || aliasedData(v, A) {
panic(ErrAliasedData)
}
if m != v.Len() {
panic(ErrDim)
}
for i := 0; i < m; i++ {
var tmp float64
for j := 0; j < n; j++ {
tmp += A.At(i, j) * b.AtVec(j)
}
v.SetVec(i, tmp)
}
}
func (v *DenseV) MulElemVec(a, b Vector) {
ar := a.Len()
if v.data == nil {
*v = *NewDenseVector(ar, nil)
}
br := b.Len()
if ar != br {
panic(ErrDim)
}
if v.Len() != ar {
panic(ErrDim)
}
for i := 0; i < ar; i++ {
v.SetVec(i, a.AtVec(i)*b.AtVec(i))
}
}
// DoSet iterates over all vector elements calling fn on them and setting
// the value at i to the result of fn.
func (A *DenseV) DoSetVec(fn func(i int, v float64) float64) {
// TODO(soypat): This could be optimized for direct access.
n := A.Len()
for i := 0; i < n; i++ {
A.SetVec(i, fn(i, A.AtVec(i)))
}
}
func aliasedData(a, b Matrix) bool {
dataA := dataHeader(a)
dataB := dataHeader(b)
if dataA.Data > dataB.Data {
// Sort in order of increasing data position in memory.
dataA, dataB = dataB, dataA
}
return dataA.Len > 0 && dataB.Len > 0 && dataA.Data+uintptr(dataA.Len) > dataB.Data
}
func dataHeader(m Matrix) reflect.SliceHeader {
var backingData []float64
switch D := m.(type) {
case *DenseM:
backingData = D.data
case *DenseV:
backingData = D.data
case SliceM:
return dataHeader(D.m)
case SliceV:
return dataHeader(D.sm)
case Transpose:
return dataHeader(D.m)
case *Sparse:
v := reflect.ValueOf(D.m)
return reflect.SliceHeader{Data: v.Pointer(), Len: 8, Cap: 8} // Best we can do?
default:
panic("unknown Matrix type. Can't determine backing data " + fmt.Sprintf("%T", D))
}
return *(*reflect.SliceHeader)(unsafe.Pointer(&backingData))
}