gnuk/src/main.c

244 lines
4.6 KiB
C
Raw Normal View History

2010-08-10 03:11:02 +00:00
/*
2010-08-30 02:39:10 +00:00
* main.c - main routine of Gnuk
*
* Copyright (C) 2010 Free Software Initiative of Japan
* Author: NIIBE Yutaka <gniibe@fsij.org>
*
* This file is a part of Gnuk, a GnuPG USB Token implementation.
*
* Gnuk 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 3 of the License, or
* (at your option) any later version.
*
* Gnuk 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, see <http://www.gnu.org/licenses/>.
*
*/
2010-08-10 03:11:02 +00:00
2010-09-04 04:48:26 +00:00
#include "config.h"
2010-08-10 03:11:02 +00:00
#include "ch.h"
#include "hal.h"
2010-08-10 06:35:34 +00:00
#include "usb_lld.h"
2010-08-26 10:50:06 +00:00
#include "gnuk.h"
2010-08-10 06:35:34 +00:00
#include "usb_lib.h"
#include "usb_istr.h"
#include "usb_desc.h"
#include "hw_config.h"
#include "usb_pwr.h"
2010-08-10 03:11:02 +00:00
2010-09-03 15:42:36 +00:00
#ifdef DEBUG
2010-08-10 07:30:05 +00:00
struct stdout {
Mutex m;
CondVar start_cnd;
CondVar finish_cnd;
const char *str;
int size;
2010-09-04 04:48:26 +00:00
};
static struct stdout stdout;
2010-08-10 07:30:05 +00:00
static void
stdout_init (void)
{
chMtxInit (&stdout.m);
chCondInit (&stdout.start_cnd);
chCondInit (&stdout.finish_cnd);
stdout.size = 0;
stdout.str = NULL;
}
2010-08-26 10:50:06 +00:00
void
2010-08-10 07:30:05 +00:00
_write (const char *s, int size)
{
2010-08-10 08:47:55 +00:00
if (size == 0)
2010-08-26 10:50:06 +00:00
return;
2010-08-10 08:47:55 +00:00
2010-08-10 07:30:05 +00:00
chMtxLock (&stdout.m);
2010-08-18 08:02:04 +00:00
while (stdout.str)
2010-08-10 07:30:05 +00:00
chCondWait (&stdout.finish_cnd);
stdout.str = s;
stdout.size = size;
chCondSignal (&stdout.start_cnd);
chCondWait (&stdout.finish_cnd);
chMtxUnlock ();
}
2010-08-10 06:35:34 +00:00
extern uint32_t count_in;
extern __IO uint32_t count_out;
extern uint8_t buffer_in[VIRTUAL_COM_PORT_DATA_SIZE];
extern uint8_t buffer_out[VIRTUAL_COM_PORT_DATA_SIZE];
2010-09-03 15:42:36 +00:00
static WORKING_AREA(waSTDOUTthread, 128);
2010-09-04 04:48:26 +00:00
static msg_t
STDOUTthread (void *arg)
2010-08-10 07:30:05 +00:00
{
(void)arg;
again:
while (1)
{
if (bDeviceState == CONFIGURED)
break;
chThdSleepMilliseconds (100);
}
while (1)
{
2010-08-10 08:47:55 +00:00
const char *p;
int len;
2010-08-10 07:30:05 +00:00
if (bDeviceState != CONFIGURED)
break;
chMtxLock (&stdout.m);
if (stdout.str == NULL)
chCondWait (&stdout.start_cnd);
2010-08-10 08:47:55 +00:00
p = stdout.str;
len = stdout.size;
while (len > 0)
{
int i;
if (len < VIRTUAL_COM_PORT_DATA_SIZE)
{
for (i = 0; i < len; i++)
buffer_in[i] = p[i];
count_in = len;
len = 0;
}
else
{
for (i = 0; i < VIRTUAL_COM_PORT_DATA_SIZE; i++)
buffer_in[i] = p[i];
len -= VIRTUAL_COM_PORT_DATA_SIZE;
count_in = VIRTUAL_COM_PORT_DATA_SIZE;
p += count_in;
}
2010-09-04 09:44:01 +00:00
USB_SIL_Write (EP3_IN, buffer_in, count_in);
SetEPTxValid (ENDP3);
2010-08-10 08:47:55 +00:00
while (count_in > 0)
chThdSleepMilliseconds (1);
}
2010-08-10 07:30:05 +00:00
stdout.str = NULL;
stdout.size = 0;
2010-08-18 08:02:04 +00:00
chCondBroadcast (&stdout.finish_cnd);
2010-08-10 07:30:05 +00:00
chMtxUnlock ();
}
goto again;
return 0;
}
2010-09-03 15:42:36 +00:00
#else
void
_write (const char *s, int size)
{
(void)s;
(void)size;
}
#endif
2010-08-10 07:30:05 +00:00
2010-09-04 04:48:26 +00:00
static WORKING_AREA(waUSBthread, 128);
2010-08-19 08:09:59 +00:00
extern msg_t USBthread (void *arg);
2010-08-23 07:44:02 +00:00
static WORKING_AREA(waGPGthread, 128*16);
2010-08-19 08:09:59 +00:00
extern msg_t GPGthread (void *arg);
2010-08-18 05:21:58 +00:00
2010-09-04 04:48:26 +00:00
Thread *blinker_thread;
2010-09-03 15:42:36 +00:00
/*
2010-09-04 04:48:26 +00:00
* Red LEDs blinker
2010-09-03 15:42:36 +00:00
*/
2010-09-04 04:48:26 +00:00
#define EV_LED (eventmask_t)1
2010-09-03 15:42:36 +00:00
2010-08-10 03:11:02 +00:00
/*
* Entry point, note, the main() function is already a thread in the system
* on entry.
*/
2010-08-30 02:39:10 +00:00
int
main (int argc, char **argv)
2010-08-10 07:30:05 +00:00
{
2010-09-04 04:48:26 +00:00
eventmask_t m;
2010-08-10 07:30:05 +00:00
int count = 0;
2010-08-10 03:11:02 +00:00
(void)argc;
(void)argv;
2010-09-04 04:48:26 +00:00
blinker_thread = chThdSelf ();
2010-08-26 10:50:06 +00:00
gpg_do_table_init ();
2010-08-10 06:35:34 +00:00
usb_lld_init ();
USB_Init();
2010-08-10 03:11:02 +00:00
2010-09-04 09:44:01 +00:00
#ifdef DEBUG
2010-08-10 07:30:05 +00:00
stdout_init ();
2010-08-10 03:11:02 +00:00
2010-08-10 07:30:05 +00:00
/*
* Creates 'stdout' thread.
*/
2010-09-04 04:48:26 +00:00
chThdCreateStatic (waSTDOUTthread, sizeof(waSTDOUTthread), NORMALPRIO, STDOUTthread, NULL);
2010-09-03 15:42:36 +00:00
#endif
2010-08-10 07:30:05 +00:00
2010-08-19 08:09:59 +00:00
chThdCreateStatic (waUSBthread, sizeof(waUSBthread), NORMALPRIO, USBthread, NULL);
chThdCreateStatic (waGPGthread, sizeof(waGPGthread), NORMALPRIO, GPGthread, NULL);
2010-08-18 05:21:58 +00:00
2010-08-10 07:30:05 +00:00
while (1)
{
2010-09-04 04:48:26 +00:00
uint8_t once = 0;
2010-08-10 08:47:55 +00:00
#if 0
2010-08-10 07:30:05 +00:00
if (palReadPad(IOPORT1, GPIOA_BUTTON))
palSetPad (IOPORT3, GPIOC_LED);
2010-08-10 08:47:55 +00:00
#endif
2010-08-10 06:35:34 +00:00
2010-09-04 04:48:26 +00:00
m = chEvtWaitOneTimeout (ALL_EVENTS, 100);
if (m == EV_LED)
palClearPad (IOPORT3, GPIOC_LED);
2010-08-10 07:30:05 +00:00
2010-09-04 04:48:26 +00:00
if (once == 0 && bDeviceState == CONFIGURED)
{
random_init ();
once = 1;
}
2010-09-03 15:42:36 +00:00
2010-09-04 04:48:26 +00:00
if (bDeviceState == CONFIGURED && (count % 100) == 0)
2010-08-10 08:47:55 +00:00
{
2010-09-03 15:42:36 +00:00
uint32_t r;
r = get_random ();
DEBUG_SHORT (r);
2010-08-10 08:47:55 +00:00
_write ("\r\nThis is ChibiOS 2.0.2 on Olimex STM32-H103.\r\n"
"Testing USB driver.\n\n"
"Hello world\r\n\r\n", 47+21+15);
}
2010-09-04 04:48:26 +00:00
m = chEvtWaitOneTimeout (ALL_EVENTS, 100);
if (m == EV_LED)
palSetPad (IOPORT3, GPIOC_LED);
2010-08-10 07:30:05 +00:00
count++;
2010-08-10 06:35:34 +00:00
}
2010-08-10 03:11:02 +00:00
return 0;
}
2010-09-03 15:42:36 +00:00
void
fatal (void)
{
_write ("fatal\r\n", 7);
for (;;);
}