package rgbmatrix /* #cgo CFLAGS: -std=c99 -I${SRCDIR}/vendor/rpi-rgb-led-matrix/include -DSHOW_REFRESH_RATE #cgo LDFLAGS: -lrgbmatrix -L${SRCDIR}/vendor/rpi-rgb-led-matrix/lib -lstdc++ -lm #include void led_matrix_swap(struct RGBLedMatrix *matrix, struct LedCanvas *offscreen_canvas, int width, int height, const uint32_t pixels[]) { int i, x, y; uint32_t color; for (x = 0; x < width; ++x) { for (y = 0; y < height; ++y) { i = x + (y * width); color = pixels[i]; led_canvas_set_pixel(offscreen_canvas, x, y, (color >> 16) & 255, (color >> 8) & 255, color & 255); } } offscreen_canvas = led_matrix_swap_on_vsync(matrix, offscreen_canvas); } void set_show_refresh_rate(struct RGBLedMatrixOptions *o, int show_refresh_rate) { o->show_refresh_rate = show_refresh_rate != 0 ? 1 : 0; } void set_disable_hardware_pulsing(struct RGBLedMatrixOptions *o, int disable_hardware_pulsing) { o->disable_hardware_pulsing = disable_hardware_pulsing != 0 ? 1 : 0; } void set_inverse_colors(struct RGBLedMatrixOptions *o, int inverse_colors) { o->inverse_colors = inverse_colors != 0 ? 1 : 0; } */ import "C" import ( "fmt" "image/color" "os" "unsafe" "github.com/mcuadros/go-rpi-rgb-led-matrix/emulator" ) // DefaultConfig default WS281x configuration var DefaultConfig = HardwareConfig{ Rows: 32, Cols: 32, ChainLength: 1, Parallel: 1, PWMBits: 11, PWMLSBNanoseconds: 130, Brightness: 100, ScanMode: Progressive, GPIOSlowdown: 1, } // HardwareConfig rgb-led-matrix configuration type HardwareConfig struct { // Rows the number of rows supported by the display, so 32 or 16. Rows int // Cols the number of columns supported by the display, so 32 or 64 . Cols int // ChainLengthis the number of displays daisy-chained together // (output of one connected to input of next). ChainLength int // Parallel is the number of parallel chains connected to the Pi; in old Pis // with 26 GPIO pins, that is 1, in newer Pis with 40 interfaces pins, that // can also be 2 or 3. The effective number of pixels in vertical direction is // then thus rows * parallel. Parallel int // Set PWM bits used for output. Default is 11, but if you only deal with // limited comic-colors, 1 might be sufficient. Lower require less CPU and // increases refresh-rate. PWMBits int // Change the base time-unit for the on-time in the lowest significant bit in // nanoseconds. Higher numbers provide better quality (more accurate color, // less ghosting), but have a negative impact on the frame rate. PWMLSBNanoseconds int // the DMA channel to use // Brightness is the initial brightness of the panel in percent. Valid range // is 1..100 Brightness int // ScanMode progressive or interlaced ScanMode ScanMode // strip color layout // Disable the PWM hardware subsystem to create pulses. Typically, you don't // want to disable hardware pulsing, this is mostly for debugging and figuring // out if there is interference with the sound system. // This won't do anything if output enable is not connected to GPIO 18 in // non-standard wirings. DisableHardwarePulsing bool ShowRefreshRate bool InverseColors bool // Control speed of GPIO updates. Valid range is 0..3 GPIOSlowdown int // Name of GPIO mapping used HardwareMapping string } func (c *HardwareConfig) geometry() (width, height int) { return c.Cols * c.ChainLength, c.Rows * c.Parallel } func (c *HardwareConfig) toC() *C.struct_RGBLedMatrixOptions { o := &C.struct_RGBLedMatrixOptions{} o.rows = C.int(c.Rows) o.cols = C.int(c.Cols) o.chain_length = C.int(c.ChainLength) o.parallel = C.int(c.Parallel) o.pwm_bits = C.int(c.PWMBits) o.pwm_lsb_nanoseconds = C.int(c.PWMLSBNanoseconds) o.brightness = C.int(c.Brightness) o.scan_mode = C.int(c.ScanMode) o.hardware_mapping = C.CString(c.HardwareMapping) if c.GPIOSlowdown >= 0 && c.GPIOSlowdown < 3 { C.gpio_slowdown(c.GPIOSlowdown) } if c.ShowRefreshRate == true { C.set_show_refresh_rate(o, C.int(1)) } else { C.set_show_refresh_rate(o, C.int(0)) } if c.DisableHardwarePulsing == true { C.set_disable_hardware_pulsing(o, C.int(1)) } else { C.set_disable_hardware_pulsing(o, C.int(0)) } if c.InverseColors == true { C.set_inverse_colors(o, C.int(1)) } else { C.set_inverse_colors(o, C.int(0)) } return o } type ScanMode int8 const ( Progressive ScanMode = 0 Interlaced ScanMode = 1 ) // RGBLedMatrix matrix representation for ws281x type RGBLedMatrix struct { Config *HardwareConfig height int width int matrix *C.struct_RGBLedMatrix buffer *C.struct_LedCanvas leds []C.uint32_t } const MatrixEmulatorENV = "MATRIX_EMULATOR" // NewRGBLedMatrix returns a new matrix using the given size and config func NewRGBLedMatrix(config *HardwareConfig) (c Matrix, err error) { defer func() { if r := recover(); r != nil { var ok bool err, ok = r.(error) if !ok { err = fmt.Errorf("error creating matrix: %v", r) } } }() if isMatrixEmulator() { return buildMatrixEmulator(config), nil } w, h := config.geometry() m := C.led_matrix_create_from_options(config.toC(), nil, nil) b := C.led_matrix_create_offscreen_canvas(m) c = &RGBLedMatrix{ Config: config, width: w, height: h, matrix: m, buffer: b, leds: make([]C.uint32_t, w*h), } if m == nil { return nil, fmt.Errorf("unable to allocate memory") } return c, nil } func isMatrixEmulator() bool { if os.Getenv(MatrixEmulatorENV) == "1" { return true } return false } func buildMatrixEmulator(config *HardwareConfig) Matrix { w, h := config.geometry() return emulator.NewEmulator(w, h, emulator.DefaultPixelPitch, true) } // Initialize initialize library, must be called once before other functions are // called. func (c *RGBLedMatrix) Initialize() error { return nil } // Geometry returns the width and the height of the matrix func (c *RGBLedMatrix) Geometry() (width, height int) { return c.width, c.height } // Apply set all the pixels to the values contained in leds func (c *RGBLedMatrix) Apply(leds []color.Color) error { for position, l := range leds { c.Set(position, l) } return c.Render() } // Render update the display with the data from the LED buffer func (c *RGBLedMatrix) Render() error { w, h := c.Config.geometry() C.led_matrix_swap( c.matrix, c.buffer, C.int(w), C.int(h), (*C.uint32_t)(unsafe.Pointer(&c.leds[0])), ) c.leds = make([]C.uint32_t, w*h) return nil } // At return an Color which allows access to the LED display data as // if it were a sequence of 24-bit RGB values. func (c *RGBLedMatrix) At(position int) color.Color { return uint32ToColor(c.leds[position]) } // Set set LED at position x,y to the provided 24-bit color value. func (c *RGBLedMatrix) Set(position int, color color.Color) { c.leds[position] = C.uint32_t(colorToUint32(color)) } // Close finalizes the ws281x interface func (c *RGBLedMatrix) Close() error { C.led_matrix_delete(c.matrix) return nil } func colorToUint32(c color.Color) uint32 { if c == nil { return 0 } // A color's RGBA method returns values in the range [0, 65535] red, green, blue, _ := c.RGBA() return (red>>8)<<16 | (green>>8)<<8 | blue>>8 } func uint32ToColor(u C.uint32_t) color.Color { return color.RGBA{ uint8(u>>16) & 255, uint8(u>>8) & 255, uint8(u>>0) & 255, 0, } }