Beat Küng
8 years ago
committed by
Lorenz Meier
23 changed files with 0 additions and 865 deletions
@ -1,42 +0,0 @@
@@ -1,42 +0,0 @@
|
||||
############################################################################ |
||||
# |
||||
# Copyright (c) 2015 PX4 Development Team. All rights reserved. |
||||
# |
||||
# Redistribution and use in source and binary forms, with or without |
||||
# modification, are permitted provided that the following conditions |
||||
# are met: |
||||
# |
||||
# 1. Redistributions of source code must retain the above copyright |
||||
# notice, this list of conditions and the following disclaimer. |
||||
# 2. Redistributions in binary form must reproduce the above copyright |
||||
# notice, this list of conditions and the following disclaimer in |
||||
# the documentation and/or other materials provided with the |
||||
# distribution. |
||||
# 3. Neither the name PX4 nor the names of its contributors may be |
||||
# used to endorse or promote products derived from this software |
||||
# without specific prior written permission. |
||||
# |
||||
# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS |
||||
# "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT |
||||
# LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS |
||||
# FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE |
||||
# COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, |
||||
# INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, |
||||
# BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS |
||||
# OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED |
||||
# AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT |
||||
# LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN |
||||
# ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE |
||||
# POSSIBILITY OF SUCH DAMAGE. |
||||
# |
||||
############################################################################ |
||||
px4_add_module( |
||||
MODULE platforms__posix__drivers__rgbledsim |
||||
MAIN rgbledsim |
||||
COMPILE_FLAGS |
||||
SRCS |
||||
rgbled.cpp |
||||
DEPENDS |
||||
platforms__common |
||||
) |
||||
# vim: set noet ft=cmake fenc=utf-8 ff=unix : |
@ -1,802 +0,0 @@
@@ -1,802 +0,0 @@
|
||||
/****************************************************************************
|
||||
* |
||||
* Copyright (c) 2012-2015 PX4 Development Team. All rights reserved. |
||||
* |
||||
* Redistribution and use in source and binary forms, with or without |
||||
* modification, are permitted provided that the following conditions |
||||
* are met: |
||||
* |
||||
* 1. Redistributions of source code must retain the above copyright |
||||
* notice, this list of conditions and the following disclaimer. |
||||
* 2. Redistributions in binary form must reproduce the above copyright |
||||
* notice, this list of conditions and the following disclaimer in |
||||
* the documentation and/or other materials provided with the |
||||
* distribution. |
||||
* 3. Neither the name PX4 nor the names of its contributors may be |
||||
* used to endorse or promote products derived from this software |
||||
* without specific prior written permission. |
||||
* |
||||
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS |
||||
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT |
||||
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS |
||||
* FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE |
||||
* COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, |
||||
* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, |
||||
* BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS |
||||
* OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED |
||||
* AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT |
||||
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN |
||||
* ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE |
||||
* POSSIBILITY OF SUCH DAMAGE. |
||||
* |
||||
****************************************************************************/ |
||||
|
||||
/**
|
||||
* @file rgbled.cpp |
||||
* |
||||
* Driver for the onboard RGB LED controller (TCA62724FMG) connected via I2C. |
||||
* |
||||
* @author Julian Oes <julian@px4.io> |
||||
* @author Anton Babushkin <anton.babushkin@me.com> |
||||
*/ |
||||
|
||||
#include <px4_config.h> |
||||
#include <px4_getopt.h> |
||||
|
||||
#include <sys/types.h> |
||||
#include <stdint.h> |
||||
#include <string.h> |
||||
#include <stdlib.h> |
||||
#include <stdbool.h> |
||||
#include <fcntl.h> |
||||
#include <unistd.h> |
||||
#include <stdio.h> |
||||
#include <ctype.h> |
||||
#include <errno.h> |
||||
|
||||
#include <px4_workqueue.h> |
||||
|
||||
#include <systemlib/perf_counter.h> |
||||
#include <systemlib/err.h> |
||||
#include <systemlib/systemlib.h> |
||||
|
||||
#include <board_config.h> |
||||
|
||||
#include <drivers/drv_rgbled.h> |
||||
|
||||
#include "VirtDevObj.hpp" |
||||
|
||||
using namespace DriverFramework; |
||||
|
||||
#define DEVICE_LOG PX4_INFO |
||||
#define RGBLED_ONTIME 120 |
||||
#define RGBLED_OFFTIME 120 |
||||
|
||||
#define ADDR PX4_I2C_OBDEV_LED /**< I2C adress of TCA62724FMG */ |
||||
#define SUB_ADDR_START 0x01 /**< write everything (with auto-increment) */ |
||||
#define SUB_ADDR_PWM0 0x81 /**< blue (without auto-increment) */ |
||||
#define SUB_ADDR_PWM1 0x82 /**< green (without auto-increment) */ |
||||
#define SUB_ADDR_PWM2 0x83 /**< red (without auto-increment) */ |
||||
#define SUB_ADDR_SETTINGS 0x84 /**< settings (without auto-increment)*/ |
||||
|
||||
#define SETTING_NOT_POWERSAVE 0x01 /**< power-save mode not off */ |
||||
#define SETTING_ENABLE 0x02 /**< on */ |
||||
|
||||
|
||||
class RGBLEDSIM : public VirtDevObj |
||||
{ |
||||
public: |
||||
RGBLEDSIM(int bus, int rgbled); |
||||
virtual ~RGBLEDSIM(); |
||||
|
||||
|
||||
virtual int init(); |
||||
virtual int probe(); |
||||
virtual int info(); |
||||
virtual int devIOCTL(unsigned long cmd, unsigned long arg); |
||||
|
||||
// Don't use the default periodic callback
|
||||
virtual void _measure() {} |
||||
|
||||
private: |
||||
work_s _work; |
||||
|
||||
rgbled_mode_t _mode; |
||||
rgbled_pattern_t _pattern; |
||||
|
||||
uint8_t _r; |
||||
uint8_t _g; |
||||
uint8_t _b; |
||||
float _brightness; |
||||
float _max_brightness; |
||||
|
||||
bool _running; |
||||
int _led_interval; |
||||
bool _should_run; |
||||
int _counter; |
||||
int _param_sub; |
||||
|
||||
void set_color(rgbled_color_t ledcolor); |
||||
void set_mode(rgbled_mode_t mode); |
||||
void set_pattern(rgbled_pattern_t *pattern); |
||||
|
||||
static void led_trampoline(void *arg); |
||||
void led(); |
||||
|
||||
int send_led_enable(bool enable); |
||||
int send_led_rgb(); |
||||
int get(bool &on, bool &powersave, uint8_t &r, uint8_t &g, uint8_t &b); |
||||
void update_params(); |
||||
|
||||
/**
|
||||
* Perform an I2C transaction to the device. |
||||
* |
||||
* At least one of send_len and recv_len must be non-zero. |
||||
* |
||||
* @param send Pointer to bytes to send. |
||||
* @param send_len Number of bytes to send. |
||||
* @param recv Pointer to buffer for bytes received. |
||||
* @param recv_len Number of bytes to receive. |
||||
* @return OK if the transfer was successful, -errno |
||||
* otherwise. |
||||
*/ |
||||
int transfer(const uint8_t *send, unsigned send_len, |
||||
uint8_t *recv, unsigned recv_len); |
||||
|
||||
}; |
||||
|
||||
/* for now, we only support one RGBLEDSIM */ |
||||
namespace |
||||
{ |
||||
RGBLEDSIM *g_rgbled = nullptr; |
||||
} |
||||
|
||||
void rgbled_usage(); |
||||
|
||||
extern "C" __EXPORT int rgbledsim_main(int argc, char *argv[]); |
||||
|
||||
RGBLEDSIM::RGBLEDSIM(int bus, int rgbled) : |
||||
VirtDevObj("rgbled", "/dev/rgbledsim", RGBLED_BASE_DEVICE_PATH, 0), |
||||
_mode(RGBLED_MODE_OFF), |
||||
_r(0), |
||||
_g(0), |
||||
_b(0), |
||||
_brightness(1.0f), |
||||
_max_brightness(1.0f), |
||||
_running(false), |
||||
_led_interval(0), |
||||
_should_run(false), |
||||
_counter(0), |
||||
_param_sub(-1) |
||||
{ |
||||
memset(&_work, 0, sizeof(_work)); |
||||
memset(&_pattern, 0, sizeof(_pattern)); |
||||
} |
||||
|
||||
RGBLEDSIM::~RGBLEDSIM() |
||||
{ |
||||
} |
||||
|
||||
int |
||||
RGBLEDSIM::init() |
||||
{ |
||||
int ret; |
||||
ret = VirtDevObj::init(); |
||||
|
||||
if (ret != OK) { |
||||
return ret; |
||||
} |
||||
|
||||
/* switch off LED on start */ |
||||
send_led_enable(false); |
||||
send_led_rgb(); |
||||
|
||||
return OK; |
||||
} |
||||
|
||||
int |
||||
RGBLEDSIM::probe() |
||||
{ |
||||
int ret; |
||||
bool on, powersave; |
||||
uint8_t r, g, b; |
||||
|
||||
/**
|
||||
this may look strange, but is needed. There is a serial |
||||
EEPROM (Microchip-24aa01) on the PX4FMU-v1 that responds to |
||||
a bunch of I2C addresses, including the 0x55 used by this |
||||
LED device. So we need to do enough operations to be sure |
||||
we are talking to the right device. These 3 operations seem |
||||
to be enough, as the 3rd one consistently fails if no |
||||
RGBLEDSIM is on the bus. |
||||
*/ |
||||
|
||||
if ((ret = get(on, powersave, r, g, b)) != OK || |
||||
(ret = send_led_enable(false) != OK) || |
||||
(ret = send_led_enable(false) != OK)) { |
||||
return ret; |
||||
} |
||||
|
||||
return ret; |
||||
} |
||||
|
||||
int |
||||
RGBLEDSIM::info() |
||||
{ |
||||
int ret; |
||||
bool on, powersave; |
||||
uint8_t r, g, b; |
||||
|
||||
ret = get(on, powersave, r, g, b); |
||||
|
||||
if (ret == OK) { |
||||
/* we don't care about power-save mode */ |
||||
DEVICE_LOG("state: %s", on ? "ON" : "OFF"); |
||||
DEVICE_LOG("red: %u, green: %u, blue: %u", (unsigned)r, (unsigned)g, (unsigned)b); |
||||
|
||||
} else { |
||||
PX4_WARN("failed to read led"); |
||||
} |
||||
|
||||
return ret; |
||||
} |
||||
|
||||
int |
||||
RGBLEDSIM::devIOCTL(unsigned long cmd, unsigned long arg) |
||||
{ |
||||
int ret = ENOTTY; |
||||
|
||||
switch (cmd) { |
||||
case RGBLED_SET_RGB: |
||||
/* set the specified color */ |
||||
_r = ((rgbled_rgbset_t *) arg)->red; |
||||
_g = ((rgbled_rgbset_t *) arg)->green; |
||||
_b = ((rgbled_rgbset_t *) arg)->blue; |
||||
send_led_rgb(); |
||||
return OK; |
||||
|
||||
case RGBLED_SET_COLOR: |
||||
/* set the specified color name */ |
||||
set_color((rgbled_color_t)arg); |
||||
send_led_rgb(); |
||||
return OK; |
||||
|
||||
case RGBLED_SET_MODE: |
||||
/* set the specified mode */ |
||||
set_mode((rgbled_mode_t)arg); |
||||
return OK; |
||||
|
||||
case RGBLED_SET_PATTERN: |
||||
/* set a special pattern */ |
||||
set_pattern((rgbled_pattern_t *)arg); |
||||
return OK; |
||||
|
||||
default: |
||||
/* see if the parent class can make any use of it */ |
||||
ret = VirtDevObj::devIOCTL(cmd, arg); |
||||
break; |
||||
} |
||||
|
||||
return ret; |
||||
} |
||||
|
||||
|
||||
void |
||||
RGBLEDSIM::led_trampoline(void *arg) |
||||
{ |
||||
RGBLEDSIM *rgbl = reinterpret_cast<RGBLEDSIM *>(arg); |
||||
|
||||
rgbl->led(); |
||||
} |
||||
|
||||
/**
|
||||
* Main loop function |
||||
*/ |
||||
void |
||||
RGBLEDSIM::led() |
||||
{ |
||||
if (!_should_run) { |
||||
_running = false; |
||||
return; |
||||
} |
||||
|
||||
if (_param_sub < 0) { |
||||
_param_sub = orb_subscribe(ORB_ID(parameter_update)); |
||||
} |
||||
|
||||
if (_param_sub >= 0) { |
||||
bool updated = false; |
||||
orb_check(_param_sub, &updated); |
||||
|
||||
if (updated) { |
||||
parameter_update_s pupdate; |
||||
orb_copy(ORB_ID(parameter_update), _param_sub, &pupdate); |
||||
update_params(); |
||||
// Immediately update to change brightness
|
||||
send_led_rgb(); |
||||
} |
||||
} |
||||
|
||||
switch (_mode) { |
||||
case RGBLED_MODE_BLINK_SLOW: |
||||
case RGBLED_MODE_BLINK_NORMAL: |
||||
case RGBLED_MODE_BLINK_FAST: |
||||
if (_counter >= 2) { |
||||
_counter = 0; |
||||
} |
||||
|
||||
send_led_enable(_counter == 0); |
||||
|
||||
break; |
||||
|
||||
case RGBLED_MODE_BREATHE: |
||||
|
||||
if (_counter >= 62) { |
||||
_counter = 0; |
||||
} |
||||
|
||||
int n; |
||||
|
||||
if (_counter < 32) { |
||||
n = _counter; |
||||
|
||||
} else { |
||||
n = 62 - _counter; |
||||
} |
||||
|
||||
_brightness = n * n / (31.0f * 31.0f); |
||||
send_led_rgb(); |
||||
break; |
||||
|
||||
case RGBLED_MODE_PATTERN: |
||||
|
||||
/* don't run out of the pattern array and stop if the next frame is 0 */ |
||||
if (_counter >= RGBLED_PATTERN_LENGTH || _pattern.duration[_counter] <= 0) { |
||||
_counter = 0; |
||||
} |
||||
|
||||
set_color(_pattern.color[_counter]); |
||||
send_led_rgb(); |
||||
_led_interval = _pattern.duration[_counter]; |
||||
break; |
||||
|
||||
default: |
||||
break; |
||||
} |
||||
|
||||
_counter++; |
||||
} |
||||
|
||||
/**
|
||||
* Parse color constant and set _r _g _b values |
||||
*/ |
||||
void |
||||
RGBLEDSIM::set_color(rgbled_color_t color) |
||||
{ |
||||
switch (color) { |
||||
case RGBLED_COLOR_OFF: |
||||
_r = 0; |
||||
_g = 0; |
||||
_b = 0; |
||||
break; |
||||
|
||||
case RGBLED_COLOR_RED: |
||||
_r = 255; |
||||
_g = 0; |
||||
_b = 0; |
||||
break; |
||||
|
||||
case RGBLED_COLOR_YELLOW: |
||||
_r = 255; |
||||
_g = 200; |
||||
_b = 0; |
||||
break; |
||||
|
||||
case RGBLED_COLOR_PURPLE: |
||||
_r = 255; |
||||
_g = 0; |
||||
_b = 255; |
||||
break; |
||||
|
||||
case RGBLED_COLOR_GREEN: |
||||
_r = 0; |
||||
_g = 255; |
||||
_b = 0; |
||||
break; |
||||
|
||||
case RGBLED_COLOR_BLUE: |
||||
_r = 0; |
||||
_g = 0; |
||||
_b = 255; |
||||
break; |
||||
|
||||
case RGBLED_COLOR_WHITE: |
||||
_r = 255; |
||||
_g = 255; |
||||
_b = 255; |
||||
break; |
||||
|
||||
case RGBLED_COLOR_AMBER: |
||||
_r = 255; |
||||
_g = 80; |
||||
_b = 0; |
||||
break; |
||||
|
||||
case RGBLED_COLOR_DIM_RED: |
||||
_r = 90; |
||||
_g = 0; |
||||
_b = 0; |
||||
break; |
||||
|
||||
case RGBLED_COLOR_DIM_YELLOW: |
||||
_r = 80; |
||||
_g = 30; |
||||
_b = 0; |
||||
break; |
||||
|
||||
case RGBLED_COLOR_DIM_PURPLE: |
||||
_r = 45; |
||||
_g = 0; |
||||
_b = 45; |
||||
break; |
||||
|
||||
case RGBLED_COLOR_DIM_GREEN: |
||||
_r = 0; |
||||
_g = 90; |
||||
_b = 0; |
||||
break; |
||||
|
||||
case RGBLED_COLOR_DIM_BLUE: |
||||
_r = 0; |
||||
_g = 0; |
||||
_b = 90; |
||||
break; |
||||
|
||||
case RGBLED_COLOR_DIM_WHITE: |
||||
_r = 30; |
||||
_g = 30; |
||||
_b = 30; |
||||
break; |
||||
|
||||
case RGBLED_COLOR_DIM_AMBER: |
||||
_r = 80; |
||||
_g = 20; |
||||
_b = 0; |
||||
break; |
||||
|
||||
default: |
||||
PX4_WARN("color unknown"); |
||||
break; |
||||
} |
||||
} |
||||
|
||||
/**
|
||||
* Set mode, if mode not changed has no any effect (doesn't reset blinks phase) |
||||
*/ |
||||
void |
||||
RGBLEDSIM::set_mode(rgbled_mode_t mode) |
||||
{ |
||||
if (mode != _mode) { |
||||
_mode = mode; |
||||
|
||||
switch (mode) { |
||||
case RGBLED_MODE_OFF: |
||||
_should_run = false; |
||||
send_led_enable(false); |
||||
break; |
||||
|
||||
case RGBLED_MODE_ON: |
||||
_brightness = 1.0f; |
||||
send_led_rgb(); |
||||
send_led_enable(true); |
||||
break; |
||||
|
||||
case RGBLED_MODE_BLINK_SLOW: |
||||
_should_run = true; |
||||
_counter = 0; |
||||
_led_interval = 2000; |
||||
_brightness = 1.0f; |
||||
send_led_rgb(); |
||||
break; |
||||
|
||||
case RGBLED_MODE_BLINK_NORMAL: |
||||
_should_run = true; |
||||
_counter = 0; |
||||
_led_interval = 500; |
||||
_brightness = 1.0f; |
||||
send_led_rgb(); |
||||
break; |
||||
|
||||
case RGBLED_MODE_BLINK_FAST: |
||||
_should_run = true; |
||||
_counter = 0; |
||||
_led_interval = 100; |
||||
_brightness = 1.0f; |
||||
send_led_rgb(); |
||||
break; |
||||
|
||||
case RGBLED_MODE_BREATHE: |
||||
_should_run = true; |
||||
_counter = 0; |
||||
_led_interval = 25; |
||||
send_led_enable(true); |
||||
break; |
||||
|
||||
case RGBLED_MODE_PATTERN: |
||||
_should_run = true; |
||||
_counter = 0; |
||||
_brightness = 1.0f; |
||||
send_led_enable(true); |
||||
break; |
||||
|
||||
default: |
||||
PX4_WARN("mode unknown"); |
||||
break; |
||||
} |
||||
|
||||
/* if it should run now, start the workq */ |
||||
if (_should_run && !_running) { |
||||
_running = true; |
||||
work_queue(LPWORK, &_work, (worker_t)&RGBLEDSIM::led_trampoline, this, 1); |
||||
} |
||||
|
||||
} |
||||
} |
||||
|
||||
/**
|
||||
* Set pattern for PATTERN mode, but don't change current mode |
||||
*/ |
||||
void |
||||
RGBLEDSIM::set_pattern(rgbled_pattern_t *pattern) |
||||
{ |
||||
memcpy(&_pattern, pattern, sizeof(rgbled_pattern_t)); |
||||
} |
||||
|
||||
/**
|
||||
* Sent ENABLE flag to LED driver |
||||
*/ |
||||
int |
||||
RGBLEDSIM::send_led_enable(bool enable) |
||||
{ |
||||
uint8_t settings_byte = 0; |
||||
|
||||
if (enable) { |
||||
settings_byte |= SETTING_ENABLE; |
||||
} |
||||
|
||||
settings_byte |= SETTING_NOT_POWERSAVE; |
||||
|
||||
const uint8_t msg[2] = { SUB_ADDR_SETTINGS, settings_byte}; |
||||
|
||||
return transfer(msg, sizeof(msg), nullptr, 0); |
||||
} |
||||
|
||||
/**
|
||||
* Send RGB PWM settings to LED driver according to current color and brightness |
||||
*/ |
||||
int |
||||
RGBLEDSIM::send_led_rgb() |
||||
{ |
||||
/* To scale from 0..255 -> 0..15 shift right by 4 bits */ |
||||
const uint8_t msg[6] = { |
||||
SUB_ADDR_PWM0, static_cast<uint8_t>((_b >> 4) * _brightness * _max_brightness + 0.5f), |
||||
SUB_ADDR_PWM1, static_cast<uint8_t>((_g >> 4) * _brightness * _max_brightness + 0.5f), |
||||
SUB_ADDR_PWM2, static_cast<uint8_t>((_r >> 4) * _brightness * _max_brightness + 0.5f) |
||||
}; |
||||
return transfer(msg, sizeof(msg), nullptr, 0); |
||||
} |
||||
|
||||
int |
||||
RGBLEDSIM::get(bool &on, bool &powersave, uint8_t &r, uint8_t &g, uint8_t &b) |
||||
{ |
||||
uint8_t result[2] = {0, 0}; |
||||
int ret; |
||||
|
||||
ret = transfer(nullptr, 0, &result[0], 2); |
||||
|
||||
if (ret == OK) { |
||||
on = result[0] & SETTING_ENABLE; |
||||
powersave = !(result[0] & SETTING_NOT_POWERSAVE); |
||||
/* XXX check, looks wrong */ |
||||
r = (result[0] & 0x0f) << 4; |
||||
g = (result[1] & 0xf0); |
||||
b = (result[1] & 0x0f) << 4; |
||||
} |
||||
|
||||
return ret; |
||||
} |
||||
|
||||
void |
||||
RGBLEDSIM::update_params() |
||||
{ |
||||
int32_t maxbrt = 15; |
||||
param_get(param_find("LED_RGB_MAXBRT"), &maxbrt); |
||||
maxbrt = maxbrt > 15 ? 15 : maxbrt; |
||||
maxbrt = maxbrt < 0 ? 0 : maxbrt; |
||||
|
||||
// A minimum of 2 "on" steps is required for breathe effect
|
||||
if (maxbrt == 1) { |
||||
maxbrt = 2; |
||||
} |
||||
|
||||
_max_brightness = maxbrt / 15.0f; |
||||
} |
||||
|
||||
void |
||||
rgbled_usage() |
||||
{ |
||||
PX4_WARN("missing command: try 'start', 'test', 'info', 'off', 'stop', 'rgb 30 40 50'"); |
||||
PX4_WARN("options:"); |
||||
PX4_WARN(" -b i2cbus (%d)", PX4_I2C_BUS_LED); |
||||
PX4_WARN(" -a addr (0x%x)", ADDR); |
||||
} |
||||
|
||||
int |
||||
rgbledsim_main(int argc, char *argv[]) |
||||
{ |
||||
int i2cdevice = -1; |
||||
int rgbledadr = ADDR; /* 7bit */ |
||||
|
||||
int ch; |
||||
|
||||
/* jump over start/off/etc and look at options first */ |
||||
int myoptind = 1; |
||||
const char *myoptarg = nullptr; |
||||
|
||||
while ((ch = px4_getopt(argc, argv, "a:b:", &myoptind, &myoptarg)) != EOF) { |
||||
switch (ch) { |
||||
case 'a': |
||||
rgbledadr = strtol(myoptarg, nullptr, 0); |
||||
break; |
||||
|
||||
case 'b': |
||||
i2cdevice = strtol(myoptarg, nullptr, 0); |
||||
break; |
||||
|
||||
default: |
||||
rgbled_usage(); |
||||
return 1; |
||||
} |
||||
} |
||||
|
||||
if (myoptind >= argc) { |
||||
rgbled_usage(); |
||||
return 1; |
||||
} |
||||
|
||||
const char *verb = argv[myoptind]; |
||||
|
||||
int ret; |
||||
|
||||
if (!strcmp(verb, "start")) { |
||||
if (g_rgbled != nullptr) { |
||||
PX4_WARN("already started"); |
||||
return 1; |
||||
} |
||||
|
||||
if (i2cdevice == -1) { |
||||
// try the external bus first
|
||||
i2cdevice = PX4_I2C_BUS_EXPANSION; |
||||
g_rgbled = new RGBLEDSIM(PX4_I2C_BUS_EXPANSION, rgbledadr); |
||||
|
||||
if (g_rgbled != nullptr && OK != g_rgbled->init()) { |
||||
delete g_rgbled; |
||||
g_rgbled = nullptr; |
||||
} |
||||
|
||||
if (g_rgbled == nullptr) { |
||||
// fall back to default bus
|
||||
if (PX4_I2C_BUS_LED == PX4_I2C_BUS_EXPANSION) { |
||||
PX4_WARN("no RGB led on bus #%d", i2cdevice); |
||||
return 1; |
||||
} |
||||
|
||||
i2cdevice = PX4_I2C_BUS_LED; |
||||
} |
||||
} |
||||
|
||||
if (g_rgbled == nullptr) { |
||||
g_rgbled = new RGBLEDSIM(i2cdevice, rgbledadr); |
||||
|
||||
if (g_rgbled == nullptr) { |
||||
PX4_WARN("new failed"); |
||||
return 1; |
||||
} |
||||
|
||||
if (OK != g_rgbled->init()) { |
||||
delete g_rgbled; |
||||
g_rgbled = nullptr; |
||||
PX4_WARN("no RGB led on bus #%d", i2cdevice); |
||||
return 1; |
||||
} |
||||
} |
||||
|
||||
return 0; |
||||
} |
||||
|
||||
/* need the driver past this point */ |
||||
if (g_rgbled == nullptr) { |
||||
PX4_WARN("not started"); |
||||
rgbled_usage(); |
||||
return 1; |
||||
} |
||||
|
||||
if (!strcmp(verb, "test")) { |
||||
DevHandle h; |
||||
DevMgr::getHandle(RGBLED0_DEVICE_PATH, h); |
||||
|
||||
if (!h.isValid()) { |
||||
PX4_WARN("Unable to open " RGBLED0_DEVICE_PATH); |
||||
return 1; |
||||
} |
||||
|
||||
rgbled_pattern_t pattern = { {RGBLED_COLOR_RED, RGBLED_COLOR_GREEN, RGBLED_COLOR_BLUE, RGBLED_COLOR_WHITE, RGBLED_COLOR_OFF, RGBLED_COLOR_OFF}, |
||||
{500, 500, 500, 500, 1000, 0 } // "0" indicates end of pattern
|
||||
}; |
||||
|
||||
ret = h.ioctl(RGBLED_SET_PATTERN, (unsigned long)&pattern); |
||||
ret = h.ioctl(RGBLED_SET_MODE, (unsigned long)RGBLED_MODE_PATTERN); |
||||
|
||||
DevMgr::releaseHandle(h); |
||||
return ret; |
||||
} |
||||
|
||||
if (!strcmp(verb, "info")) { |
||||
g_rgbled->info(); |
||||
return 0; |
||||
} |
||||
|
||||
if (!strcmp(verb, "off") || !strcmp(verb, "stop")) { |
||||
DevHandle h; |
||||
DevMgr::getHandle(RGBLED0_DEVICE_PATH, h); |
||||
|
||||
if (!h.isValid()) { |
||||
PX4_WARN("Unable to open " RGBLED0_DEVICE_PATH); |
||||
return 1; |
||||
} |
||||
|
||||
ret = h.ioctl(RGBLED_SET_MODE, (unsigned long)RGBLED_MODE_OFF); |
||||
DevMgr::releaseHandle(h); |
||||
|
||||
/* delete the rgbled object if stop was requested, in addition to turning off the LED. */ |
||||
if (!strcmp(verb, "stop")) { |
||||
delete g_rgbled; |
||||
g_rgbled = nullptr; |
||||
return 0; |
||||
} |
||||
|
||||
return ret; |
||||
} |
||||
|
||||
if (!strcmp(verb, "rgb")) { |
||||
if (argc < 5) { |
||||
PX4_WARN("Usage: rgbled rgb <red> <green> <blue>"); |
||||
return 1; |
||||
} |
||||
|
||||
DevHandle h; |
||||
DevMgr::getHandle(RGBLED0_DEVICE_PATH, h); |
||||
|
||||
if (!h.isValid()) { |
||||
PX4_WARN("Unable to open " RGBLED0_DEVICE_PATH); |
||||
return 1; |
||||
} |
||||
|
||||
rgbled_rgbset_t v; |
||||
v.red = strtol(argv[2], nullptr, 0); |
||||
v.green = strtol(argv[3], nullptr, 0); |
||||
v.blue = strtol(argv[4], nullptr, 0); |
||||
ret = h.ioctl(RGBLED_SET_RGB, (unsigned long)&v); |
||||
ret = h.ioctl(RGBLED_SET_MODE, (unsigned long)RGBLED_MODE_ON); |
||||
DevMgr::releaseHandle(h); |
||||
return ret; |
||||
} |
||||
|
||||
rgbled_usage(); |
||||
return 1; |
||||
} |
||||
|
||||
int RGBLEDSIM::transfer(const uint8_t *send, unsigned send_len, uint8_t *recv, unsigned recv_len) |
||||
{ |
||||
return 0; |
||||
} |
Loading…
Reference in new issue