2013-12-13 12:53:27 +00:00
|
|
|
package utils
|
|
|
|
|
2014-08-11 09:11:24 +00:00
|
|
|
// GaloisField encapsulates galois field arithmetics
|
2013-12-13 12:53:27 +00:00
|
|
|
type GaloisField struct {
|
2016-12-26 21:16:28 +00:00
|
|
|
Size int
|
|
|
|
Base int
|
2013-12-13 12:53:27 +00:00
|
|
|
ALogTbl []int
|
|
|
|
LogTbl []int
|
|
|
|
}
|
|
|
|
|
2016-12-26 21:16:28 +00:00
|
|
|
// NewGaloisField creates a new galois field
|
|
|
|
func NewGaloisField(pp, fieldSize, b int) *GaloisField {
|
2013-12-13 12:53:27 +00:00
|
|
|
result := new(GaloisField)
|
|
|
|
|
2016-12-26 21:16:28 +00:00
|
|
|
result.Size = fieldSize
|
|
|
|
result.Base = b
|
|
|
|
result.ALogTbl = make([]int, fieldSize)
|
|
|
|
result.LogTbl = make([]int, fieldSize)
|
2013-12-13 12:53:27 +00:00
|
|
|
|
|
|
|
x := 1
|
2016-12-26 21:16:28 +00:00
|
|
|
for i := 0; i < fieldSize; i++ {
|
2013-12-13 12:53:27 +00:00
|
|
|
result.ALogTbl[i] = x
|
|
|
|
x = x * 2
|
2016-12-26 21:16:28 +00:00
|
|
|
if x >= fieldSize {
|
|
|
|
x = (x ^ pp) & (fieldSize - 1)
|
2013-12-13 12:53:27 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-12-26 21:16:28 +00:00
|
|
|
for i := 0; i < fieldSize; i++ {
|
2013-12-13 12:53:27 +00:00
|
|
|
result.LogTbl[result.ALogTbl[i]] = int(i)
|
|
|
|
}
|
|
|
|
|
|
|
|
return result
|
|
|
|
}
|
|
|
|
|
2015-02-25 13:08:58 +00:00
|
|
|
func (gf *GaloisField) Zero() *GFPoly {
|
2016-12-26 21:16:28 +00:00
|
|
|
return NewGFPoly(gf, []int{0})
|
2015-02-25 13:08:58 +00:00
|
|
|
}
|
|
|
|
|
2014-08-11 09:11:24 +00:00
|
|
|
// AddOrSub add or substract two numbers
|
2013-12-13 12:53:27 +00:00
|
|
|
func (gf *GaloisField) AddOrSub(a, b int) int {
|
|
|
|
return a ^ b
|
|
|
|
}
|
|
|
|
|
2014-08-11 09:11:24 +00:00
|
|
|
// Multiply multiplys two numbers
|
2013-12-13 12:53:27 +00:00
|
|
|
func (gf *GaloisField) Multiply(a, b int) int {
|
|
|
|
if a == 0 || b == 0 {
|
|
|
|
return 0
|
|
|
|
}
|
2016-12-26 21:16:28 +00:00
|
|
|
return gf.ALogTbl[(gf.LogTbl[a]+gf.LogTbl[b])%(gf.Size-1)]
|
2013-12-13 12:53:27 +00:00
|
|
|
}
|
|
|
|
|
2014-08-11 09:11:24 +00:00
|
|
|
// Divide divides two numbers
|
2013-12-13 12:53:27 +00:00
|
|
|
func (gf *GaloisField) Divide(a, b int) int {
|
|
|
|
if b == 0 {
|
|
|
|
panic("divide by zero")
|
|
|
|
} else if a == 0 {
|
|
|
|
return 0
|
|
|
|
}
|
2016-12-26 21:16:28 +00:00
|
|
|
return gf.ALogTbl[(gf.LogTbl[a]-gf.LogTbl[b])%(gf.Size-1)]
|
2013-12-13 12:53:27 +00:00
|
|
|
}
|
2015-02-25 13:08:58 +00:00
|
|
|
|
|
|
|
func (gf *GaloisField) Invers(num int) int {
|
2016-12-26 21:16:28 +00:00
|
|
|
return gf.ALogTbl[(gf.Size-1)-gf.LogTbl[num]]
|
2015-02-25 13:08:58 +00:00
|
|
|
}
|