Files
recycled-ni-neutrino/src/driver/simple_display.cpp
Stefan Seyfried 33e3769bd7 simple_display: implement azbox me VFD support
Origin commit data
------------------
Branch: ni/coolstream
Commit: 9266b8ffbe
Author: Stefan Seyfried <seife@tuxbox-git.slipkontur.de>
Date: 2012-06-23 (Sat, 23 Jun 2012)


------------------
No further description and justification available within origin commit message!

------------------
This commit was generated by Migit
2012-06-27 21:18:19 +02:00

421 lines
7.8 KiB
C++

/*
Routines to drive simple one-line text or SPARK's 4 digit LED display
(C) 2012 Stefan Seyfried
License: GPL
This program is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation; either version 2 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program; if not, write to the Free Software
Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
*/
#ifdef HAVE_CONFIG_H
#include <config.h>
#endif
#include <driver/lcdd.h>
#include <driver/framebuffer.h>
#include <global.h>
#include <neutrino.h>
#include <fcntl.h>
#include <time.h>
#include <unistd.h>
//#include <math.h>
#include <sys/stat.h>
#if HAVE_SPARK_HARDWARE
#include "spark_led.h"
#define DISPLAY_DEV "/dev/vfd"
#define DISPLAY_LEN 4
#endif
#if HAVE_AZBOX_HARDWARE
#define DISPLAY_DEV "/proc/vfd"
#define LED_DEV "/proc/led"
#define DISPLAY_LEN 8
#endif
static char volume = 0;
//static char percent = 0;
static bool power = true;
static bool muted = false;
static bool showclock = true;
static time_t last_display = 0;
static char display_text[64] = { 0 };
static bool led_r = false;
static bool led_g = false;
static inline int dev_open()
{
int fd = open(DISPLAY_DEV, O_RDWR);
if (fd < 0)
fprintf(stderr, "[neutrino] simple_display: open " DISPLAY_DEV ": %m\n");
return fd;
}
#if HAVE_AZBOX_HARDWARE
static inline int led_open()
{
int fd = open(LED_DEV, O_RDWR);
if (fd < 0)
fprintf(stderr, "[neutrino] simple_display: open " LED_DEV ": %m\n");
return fd;
}
#endif
static void display(const char *s, bool update_timestamp = true)
{
int fd = dev_open();
int len = strlen(s);
if (fd < 0)
return;
printf("%s '%s'\n", __func__, s);
write(fd, s, len);
close(fd);
if (update_timestamp)
{
last_display = time(NULL);
/* increase timeout to ensure that everything is displayed
* the driver displays 5 characters per second */
if (len > DISPLAY_LEN)
last_display += (len - 4) / 5;
}
}
CLCD::CLCD()
{
/* do not show menu in neutrino... */
has_lcd = false;
}
CLCD* CLCD::getInstance()
{
static CLCD* lcdd = NULL;
if (lcdd == NULL)
lcdd = new CLCD();
return lcdd;
}
void CLCD::wake_up()
{
}
void* CLCD::TimeThread(void *)
{
while(1) {
sleep(1);
CLCD::getInstance()->showTime();
/* hack, just if we missed the blit() somewhere
* this will update the framebuffer once per second */
if (getenv("SPARK_NOBLIT") == NULL)
CFrameBuffer::getInstance()->blit();
}
return NULL;
}
void CLCD::init(const char *, const char *, const char *, const char *, const char *, const char *)
{
setMode(MODE_TVRADIO);
if (pthread_create (&thrTime, NULL, TimeThread, NULL) != 0 ) {
perror("[neutino] CLCD::init pthread_create(TimeThread)");
return ;
}
}
void CLCD::setlcdparameter(void)
{
}
void CLCD::showServicename(std::string, bool)
{
}
#if HAVE_SPARK_HARDWARE
void CLCD::setled(int red, int green)
{
struct aotom_ioctl_data d;
int leds[2] = { red, green };
int i;
int fd = dev_open();
if (fd < 0)
return;
printf("%s red:%d green:%d\n", __func__, red, green);
for (i = 0; i < 2; i++)
{
if (leds[i] == -1)
continue;
d.u.led.led_nr = i;
d.u.led.on = leds[i];
if (ioctl(fd, VFDSETLED, &d) < 0)
fprintf(stderr, "[neutrino] %s setled VFDSETLED: %m\n", __func__);
}
close(fd);
}
#elif HAVE_AZBOX_HARDWARE
void CLCD::setled(int red, int green)
{
static unsigned char col = '0'; /* need to remember the state. 1 == blue, 2 == red */
int leds[3] = { -1, green, red };
int i;
char s[3];
int fd = led_open();
if (fd < 0)
return;
for (i = 1; i <= 2; i++)
{
if (leds[i] == -1) /* don't touch */
continue;
col &= ~(i); /* clear the bit... */
if (leds[i])
col |= i; /* ...and set it again */
}
sprintf(s, "%c\n", col);
write(fd, s, 3);
close(fd);
//printf("%s(%d, %d): %c\n", __func__, red, green, col);
}
#else
void CLCD::setled(int /*red*/, int /*green*/)
{
}
#endif
void CLCD::showTime(bool force)
{
static bool blink = false;
int red = -1, green = -1;
if (mode == MODE_SHUTDOWN)
{
setled(1, 1);
return;
}
time_t now = time(NULL);
if (display_text[0])
{
display(display_text);
display_text[0] = '\0';
}
else if (power && (force || (showclock && (now - last_display) > 4)))
{
char timestr[DISPLAY_LEN];
struct tm *t;
static int hour = 0, minute = 0;
t = localtime(&now);
if (force || last_display || (hour != t->tm_hour) || (minute != t->tm_min)) {
hour = t->tm_hour;
minute = t->tm_min;
if (DISPLAY_LEN < 5)
sprintf(timestr, "%02d%02d", hour, minute);
else /* pad with spaces on the left side to center the time string */
sprintf(timestr, "%*s%02d:%02d",(DISPLAY_LEN - 5)/2, "", hour, minute);
display(timestr, false);
last_display = 0;
}
}
if (led_r)
red = blink;
blink = !blink;
if (led_g)
green = blink;
if (led_r || led_g)
setled(red, green);
}
void CLCD::showRCLock(int)
{
}
/* update is default true, the mute code sets it to false
* to force an update => inverted logic! */
void CLCD::showVolume(const char vol, const bool update)
{
char s[32];
const int type = (DISPLAY_LEN < 5);
const char *vol_fmt[] = { "Vol:%3d%%", "%4d" };
const char *mutestr[] = { "Vol:MUTE", "mute" };
if (vol == volume && update)
return;
volume = vol;
/* char is unsigned, so vol is never < 0 */
if (volume > 100)
volume = 100;
if (muted)
strcpy(s, mutestr[type]);
else
sprintf(s, vol_fmt[type], volume);
display(s);
}
void CLCD::showPercentOver(const unsigned char /*perc*/, const bool /*perform_update*/, const MODES)
{
}
void CLCD::showMenuText(const int, const char *text, const int, const bool)
{
if (mode != MODE_MENU_UTF8)
return;
strncpy(display_text, text, sizeof(display_text) - 1);
display_text[sizeof(display_text) - 1] = '\0';
}
void CLCD::showAudioTrack(const std::string &, const std::string & /*title*/, const std::string &)
{
if (mode != MODE_AUDIO)
return;
// ShowText(title.c_str());
}
void CLCD::showAudioPlayMode(AUDIOMODES)
{
}
void CLCD::showAudioProgress(const char, bool)
{
}
void CLCD::setMode(const MODES m, const char * const)
{
mode = m;
switch (m) {
case MODE_TVRADIO:
setled(0, 0);
showclock = true;
power = true;
showTime();
break;
case MODE_SHUTDOWN:
showclock = false;
Clear();
break;
case MODE_STANDBY:
setled(0, 1);
showclock = true;
showTime(true);
break;
default:
showclock = true;
showTime();
}
}
void CLCD::setBrightness(int)
{
}
int CLCD::getBrightness()
{
return 0;
}
void CLCD::setBrightnessStandby(int)
{
}
int CLCD::getBrightnessStandby()
{
return 0;
}
void CLCD::setPower(int)
{
}
int CLCD::getPower()
{
return 0;
}
void CLCD::togglePower(void)
{
power = !power;
if (!power)
Clear();
else
showTime(true);
}
void CLCD::setMuted(bool mu)
{
printf("spark_led:%s %d\n", __func__, mu);
muted = mu;
showVolume(volume, false);
}
void CLCD::resume()
{
}
void CLCD::pause()
{
}
void CLCD::Lock()
{
}
void CLCD::Unlock()
{
}
#if HAVE_SPARK_HARDWARE
void CLCD::Clear()
{
int fd = dev_open();
if (fd < 0)
return;
int ret = ioctl(fd, VFDDISPLAYCLR);
if(ret < 0)
perror("[neutrino] spark_led Clear() VFDDISPLAYCLR");
close(fd);
printf("spark_led:%s\n", __func__);
}
#endif
#if HAVE_AZBOX_HARDWARE
void CLCD::Clear()
{
display(" ", false);
}
#endif
void CLCD::ShowIcon(vfd_icon i, bool on)
{
switch (i)
{
case VFD_ICON_CAM1:
led_r = on;
setled(led_r, -1); /* switch instant on / switch off if disabling */
break;
case VFD_ICON_PLAY:
led_g = on;
setled(-1, led_g);
break;
default:
break;
}
}
void CLCD::setEPGTitle(const std::string)
{
}