optimizer.go 2.1 KB
Newer Older
1 2 3
package pserver

/*
D
dzhwinter 已提交
4 5
// TODO(zhihong): move compile flags to cmake go_library
#cgo pkg-config: protobuf
D
dzhwinter 已提交
6
#cgo CFLAGS: -I ../../
D
dongzhihong 已提交
7
#cgo LDFLAGS: /Users/dzh/.go/src/github.com/PaddlePaddle/Paddle/build/go/pserver/cclient/libpaddle_go_optimizer.a
D
dzhwinter 已提交
8
#include "paddle/optimizer/optimizer.h"
9 10 11 12 13 14 15 16 17 18
*/
import "C"
import (
	"fmt"
	"unsafe"
)

var nullPtr = unsafe.Pointer(uintptr(0))

type optimizer struct {
19
	opt *C.struct_paddle_optimizer
D
dongzhihong 已提交
20 21 22 23 24 25 26 27 28 29 30 31 32 33 34
	// used in GetParam, reconstruct Parameter from optimizer
	ElementType ElementType
}

func cArrayToSlice(p unsafe.Pointer, len int) []byte {
	if p == nullPtr {
		return nil
	}

	// create a Go clice backed by a C array, reference:
	// https://github.com/golang/go/wiki/cgo#turning-c-arrays-into-go-slices
	//
	// Go garbage collector will not interact with this data, need
	// to be freed properly.
	return (*[1 << 30]byte)(p)[:len:len]
35 36
}

D
dzhwinter 已提交
37
func newOptimizer(paramWithConfigs ParameterWithConfig) *optimizer {
38
	o := &optimizer{}
D
dzhwinter 已提交
39 40
	p := paramWithConfigs.Param
	c := paramWithConfigs.Config
D
dongzhihong 已提交
41 42
	buffer := &p.Content[0]
	o.opt = C.paddle_create_optimizer(C.uchar(c), C.int(len(c)), unsafe.Pointer(buffer), C.int(len(p.Content)), nullPtr, 0)
43 44 45
	return o
}

D
dongzhihong 已提交
46 47 48 49 50 51
func (o *optimizer) GetWeights(p *Parameter) error {

	var buffer unsafe.Pointer
	buffer_len := C.paddle_optimizer_get_weights(unsafe.Pointer(o), &buffer)
	if buffer_len == 0 || buffer == nullPtr {
		return fmt.Errorf("parameter optimizer error : %s get failed", p.name)
52
	}
D
dongzhihong 已提交
53 54 55
	p.Content = cArrayToSlice(buffer, int(buffer_len))
	return nil
}
56

D
dongzhihong 已提交
57 58 59
func (o *optimizer) UpdateParameter(g Gradient) error {
	if o.ElementType != g.ElementType {
		return fmt.Errorf("Name: %s, parameter and gradient element type not match, parameter: %v, gradient: %v", g.Name, g.ElementType, g.ElementType)
60 61
	}

D
dongzhihong 已提交
62 63
	// FIXME: do we need a copy? discard g.Content by GC ok
	r := C.paddle_update_parameter(o.opt, C.paddle_element_type(g.ElementType), unsafe.Pointer(g.Content), C.int(len(g.Content)))
64
	if r != 0 {
H
Helin Wang 已提交
65
		return fmt.Errorf("optimizer update returned error code: %d", r)
66 67 68 69 70 71 72
	}
	return nil
}

func (o *optimizer) Cleanup() {
	if unsafe.Pointer(o.opt) != nullPtr {
		C.paddle_release_optimizer(o.opt)
73
		o.opt = (*C.struct_paddle_optimizer)(nullPtr)
74 75
	}
}