2012-05-19 14:09:46 +00:00
|
|
|
/***************************************************************************
|
|
|
|
* __________ __ ___.
|
|
|
|
* Open \______ \ ____ ____ | | _\_ |__ _______ ___
|
|
|
|
* Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
|
|
|
|
* Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
|
|
|
|
* Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
|
|
|
|
* \/ \/ \/ \/ \/
|
|
|
|
* $Id$
|
|
|
|
*
|
|
|
|
* Copyright (C) 2012 by Amaury Pouly
|
|
|
|
*
|
|
|
|
* 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 software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
|
|
|
|
* KIND, either express or implied.
|
|
|
|
*
|
|
|
|
****************************************************************************/
|
|
|
|
#include "button-target.h"
|
|
|
|
#include "system.h"
|
|
|
|
#include "system-target.h"
|
|
|
|
#include "pinctrl-imx233.h"
|
|
|
|
#include "power-imx233.h"
|
|
|
|
#include "string.h"
|
|
|
|
#include "usb.h"
|
|
|
|
#include "backlight.h"
|
|
|
|
#include "mpr121.h"
|
|
|
|
|
|
|
|
static struct mpr121_config_t config =
|
|
|
|
{
|
|
|
|
.ele =
|
|
|
|
{
|
|
|
|
[0] = {.tth = 5, .rth = 4 },
|
|
|
|
[1] = {.tth = 5, .rth = 4 },
|
|
|
|
[2] = {.tth = 5, .rth = 4 },
|
|
|
|
[3] = {.tth = 5, .rth = 4 },
|
|
|
|
[4] = {.tth = 4, .rth = 3 },
|
|
|
|
[5] = {.tth = 4, .rth = 3 },
|
|
|
|
[6] = {.tth = 5, .rth = 4 },
|
|
|
|
[7] = {.tth = 5, .rth = 4 },
|
|
|
|
[8] = {.gpio = ELE_GPIO_OUTPUT_OPEN_LED },
|
|
|
|
},
|
|
|
|
.filters =
|
|
|
|
{
|
|
|
|
.ele =
|
|
|
|
{
|
|
|
|
.rising = {.mhd = 1, .nhd = 1, .ncl = 1, .fdl = 1 },
|
|
|
|
.falling = {.mhd = 1, .nhd = 1, .ncl = 0xff, .fdl = 2 },
|
|
|
|
}
|
|
|
|
},
|
|
|
|
.autoconf =
|
|
|
|
{
|
|
|
|
.en = true, .ren = true, .retry = RETRY_NEVER,
|
|
|
|
.usl = 0xc4, .lsl = 0x7f, .tl = 0xb0
|
|
|
|
},
|
|
|
|
.ele_en = ELE_EN0_x(7),
|
|
|
|
.cal_lock = CL_TRACK
|
|
|
|
};
|
|
|
|
|
|
|
|
#define MPR121_INTERRUPT 1
|
|
|
|
|
|
|
|
static int touchpad_btns = 0;
|
|
|
|
static long mpr121_stack[DEFAULT_STACK_SIZE/sizeof(long)];
|
|
|
|
static const char mpr121_thread_name[] = "mpr121";
|
|
|
|
static struct event_queue mpr121_queue;
|
|
|
|
|
|
|
|
static void mpr121_irq_cb(int bank, int pin)
|
|
|
|
{
|
|
|
|
(void) bank;
|
|
|
|
(void) pin;
|
|
|
|
/* the callback will not be fired until interrupt is enabled back so
|
|
|
|
* the queue will not overflow or contain multiple MPR121_INTERRUPT events */
|
|
|
|
queue_post(&mpr121_queue, MPR121_INTERRUPT, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void mpr121_thread(void)
|
|
|
|
{
|
|
|
|
struct queue_event ev;
|
|
|
|
|
|
|
|
while(1)
|
|
|
|
{
|
|
|
|
queue_wait(&mpr121_queue, &ev);
|
|
|
|
/* handle usb connect and ignore all messages except rmi interrupts */
|
|
|
|
if(ev.id == SYS_USB_CONNECTED)
|
|
|
|
{
|
|
|
|
usb_acknowledge(SYS_USB_CONNECTED_ACK);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
else if(ev.id != MPR121_INTERRUPT)
|
|
|
|
continue;
|
|
|
|
/* clear interrupt and get status */
|
|
|
|
unsigned status;
|
|
|
|
touchpad_btns = 0;
|
|
|
|
if(!mpr121_get_touch_status(&status))
|
|
|
|
{
|
|
|
|
/* ELE3: up
|
|
|
|
* ELE4: back
|
|
|
|
* ELE5: menu
|
|
|
|
* ELE6: down
|
|
|
|
* ELE7: play */
|
|
|
|
if(status & 0x8) touchpad_btns |= BUTTON_UP;
|
|
|
|
if(status & 0x10) touchpad_btns |= BUTTON_BACK;
|
|
|
|
if(status & 0x20) touchpad_btns |= BUTTON_MENU;
|
|
|
|
if(status & 0x40) touchpad_btns |= BUTTON_DOWN;
|
|
|
|
if(status & 0x80) touchpad_btns |= BUTTON_PLAY;
|
|
|
|
}
|
|
|
|
/* enable interrupt */
|
|
|
|
imx233_setup_pin_irq(0, 18, true, true, false, &mpr121_irq_cb);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* B0P18 is #IRQ line of the touchpad */
|
|
|
|
void button_init_device(void)
|
|
|
|
{
|
|
|
|
mpr121_init(0xb4);
|
|
|
|
mpr121_soft_reset();
|
|
|
|
mpr121_set_config(&config);
|
|
|
|
|
|
|
|
queue_init(&mpr121_queue, true);
|
|
|
|
create_thread(mpr121_thread, mpr121_stack, sizeof(mpr121_stack), 0,
|
|
|
|
mpr121_thread_name IF_PRIO(, PRIORITY_USER_INTERFACE) IF_COP(, CPU));
|
|
|
|
/* enable interrupt */
|
2012-05-22 20:59:06 +00:00
|
|
|
imx233_pinctrl_acquire_pin(0, 18, "mpr121 int");
|
2012-05-19 14:09:46 +00:00
|
|
|
imx233_set_pin_function(0, 18, PINCTRL_FUNCTION_GPIO);
|
|
|
|
imx233_enable_gpio_output(0, 18, false);
|
|
|
|
imx233_setup_pin_irq(0, 18, true, true, false, &mpr121_irq_cb);
|
2012-05-22 20:59:06 +00:00
|
|
|
/* hold button */
|
|
|
|
imx233_pinctrl_acquire_pin(0, 4, "hold");
|
|
|
|
imx233_set_pin_function(0, 4, PINCTRL_FUNCTION_GPIO);
|
|
|
|
imx233_enable_gpio_output(0, 4, false);
|
|
|
|
/* volume down button */
|
2012-05-22 21:41:21 +00:00
|
|
|
imx233_pinctrl_acquire_pin(2, 7, "volume down");
|
|
|
|
imx233_set_pin_function(2, 7, PINCTRL_FUNCTION_GPIO);
|
|
|
|
imx233_enable_gpio_output(2, 7, false);
|
2012-05-22 20:59:06 +00:00
|
|
|
/* volume up button */
|
2012-05-22 21:41:21 +00:00
|
|
|
imx233_pinctrl_acquire_pin(2, 8, "volume up");
|
|
|
|
imx233_set_pin_function(2, 8, PINCTRL_FUNCTION_GPIO);
|
|
|
|
imx233_enable_gpio_output(2, 8, false);
|
2012-05-19 14:09:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool button_hold(void)
|
|
|
|
{
|
|
|
|
/* B0P04: #hold */
|
|
|
|
return !imx233_get_gpio_input_mask(0, 0x10);
|
|
|
|
}
|
|
|
|
|
|
|
|
int button_read_device(void)
|
|
|
|
{
|
|
|
|
/* since sliding hold will usually trigger power, ignore power button
|
|
|
|
* for one second after hold is released */
|
|
|
|
static int power_ignore_counter = 0;
|
|
|
|
static bool old_hold;
|
|
|
|
/* light handling */
|
|
|
|
bool hold = button_hold();
|
|
|
|
if(hold != old_hold)
|
|
|
|
{
|
|
|
|
old_hold = hold;
|
|
|
|
#ifndef BOOTLOADER
|
|
|
|
backlight_hold_changed(hold);
|
|
|
|
#endif /* BOOTLOADER */
|
|
|
|
if(!hold)
|
|
|
|
power_ignore_counter = HZ;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(power_ignore_counter)
|
|
|
|
power_ignore_counter--;
|
|
|
|
|
|
|
|
int res = 0;
|
|
|
|
/* B2P07: #volume-
|
|
|
|
* B2P08: #volume+
|
|
|
|
* PSWITCH: power */
|
|
|
|
uint32_t mask = imx233_get_gpio_input_mask(2, 0x180);
|
|
|
|
if(!(mask & 0x80))
|
|
|
|
res |= BUTTON_VOL_DOWN;
|
|
|
|
if(!(mask & 0x100))
|
|
|
|
res |= BUTTON_VOL_UP;
|
|
|
|
/* WARNING: it seems that the voltage on PSWITCH depends on whether
|
|
|
|
* usb is connected or not ! Thus the value of this field can be 1 or 3 */
|
|
|
|
if(__XTRACT(HW_POWER_STS, PSWITCH) != 0 && power_ignore_counter == 0)
|
|
|
|
res |= BUTTON_POWER;
|
|
|
|
return res | touchpad_btns;
|
|
|
|
}
|