You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
308 lines
6.0 KiB
308 lines
6.0 KiB
13 years ago
|
/*
|
||
|
* carlu - userspace testing utility for ar9170 devices
|
||
|
*
|
||
|
* main program routine
|
||
|
*
|
||
|
* Copyright 2009-2011 Christian Lamparter <chunkeey@googlemail.com>
|
||
|
*
|
||
|
* 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 <stdio.h>
|
||
|
#include <stdbool.h>
|
||
|
#include <stdlib.h>
|
||
|
#include <errno.h>
|
||
|
#include <string.h>
|
||
|
#include <stdlib.h>
|
||
|
#include "SDL.h"
|
||
|
#include <SDL_version.h>
|
||
|
|
||
|
#include "debug.h"
|
||
|
#include "carlu.h"
|
||
|
#include "usb.h"
|
||
|
#include "frame.h"
|
||
|
#include "test.h"
|
||
|
#include "cmd.h"
|
||
|
|
||
|
void *carlu_alloc_driver(size_t size)
|
||
|
{
|
||
|
unsigned int i;
|
||
|
struct carlu *ar;
|
||
|
|
||
|
if (size < sizeof(*ar)) {
|
||
|
err("bogus driver context request.");
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
ar = malloc(size);
|
||
|
if (ar == NULL) {
|
||
|
err("failed to alloc driver context.");
|
||
|
return NULL;
|
||
|
}
|
||
|
memset(ar, 0, size);
|
||
|
|
||
|
for (i = 0; i < __AR9170_NUM_TXQ; i++)
|
||
|
frame_queue_init(&ar->tx_sent_queue[i]);
|
||
|
ar->resp_lock = SDL_CreateMutex();
|
||
|
ar->mem_lock = SDL_CreateMutex();
|
||
|
ar->resp_pend = SDL_CreateCond();
|
||
|
ar->tx_pending = 0;
|
||
|
return ar;
|
||
|
}
|
||
|
|
||
|
void carlu_free_driver(struct carlu *ar)
|
||
|
{
|
||
|
unsigned int i;
|
||
|
|
||
|
dbg("destroy driver struct.\n");
|
||
|
SDL_DestroyMutex(ar->resp_lock);
|
||
|
SDL_DestroyMutex(ar->mem_lock);
|
||
|
SDL_DestroyCond(ar->resp_pend);
|
||
|
|
||
|
for (i = 0; i < __AR9170_NUM_TXQ; i++)
|
||
|
frame_queue_kill(&ar->tx_sent_queue[i]);
|
||
|
|
||
|
free(ar);
|
||
|
}
|
||
|
|
||
|
static int carlu_init()
|
||
|
{
|
||
|
struct SDL_version compiled;
|
||
|
int ret;
|
||
|
|
||
|
SDL_VERSION(&compiled);
|
||
|
dbg("=== SDL %d.%d.%d ===\n", compiled.major, compiled.minor, compiled.patch);
|
||
|
|
||
|
ret = SDL_Init(SDL_INIT_TIMER);
|
||
|
if (ret != 0) {
|
||
|
err("Unable to initialize SDL: (%s)\n", SDL_GetError());
|
||
|
return EXIT_FAILURE;
|
||
|
}
|
||
|
|
||
|
return usb_init();
|
||
|
}
|
||
|
|
||
|
static void carlu_exit()
|
||
|
{
|
||
|
SDL_Quit();
|
||
|
usb_exit();
|
||
|
}
|
||
|
|
||
|
static int carlu_dump_eeprom(void)
|
||
|
{
|
||
|
struct carlu *carl = NULL;
|
||
|
uint8_t data[8192] = { 0 };
|
||
|
int err;
|
||
|
|
||
|
err = carlu_init();
|
||
|
if (err)
|
||
|
goto out;
|
||
|
|
||
|
carl = carlusb_probe();
|
||
|
if (IS_ERR_OR_NULL(carl)) {
|
||
|
err = PTR_ERR(carl);
|
||
|
goto out;
|
||
|
}
|
||
|
|
||
|
err = carlu_cmd_mem_dump(carl, 0, sizeof(data), &data);
|
||
|
if (err)
|
||
|
goto out_close;
|
||
|
|
||
|
print_hex_dump_bytes(INFO, "EEPROM:", data, sizeof(data));
|
||
|
|
||
|
out_close:
|
||
|
carlusb_close(carl);
|
||
|
|
||
|
out:
|
||
|
carlu_exit();
|
||
|
return err ? EXIT_FAILURE : EXIT_SUCCESS;
|
||
|
}
|
||
|
|
||
|
static int carlu_run_gpio_test(void)
|
||
|
{
|
||
|
struct carlu *carl = NULL;
|
||
|
int err;
|
||
|
|
||
|
err = carlu_init();
|
||
|
if (err)
|
||
|
goto out;
|
||
|
|
||
|
carl = carlusb_probe();
|
||
|
if (IS_ERR_OR_NULL(carl)) {
|
||
|
err = PTR_ERR(carl);
|
||
|
goto out;
|
||
|
}
|
||
|
|
||
|
err = carlu_gpio_test(carl);
|
||
|
if (err)
|
||
|
goto out_close;
|
||
|
|
||
|
out_close:
|
||
|
carlusb_close(carl);
|
||
|
|
||
|
out:
|
||
|
carlu_exit();
|
||
|
return err ? EXIT_FAILURE : EXIT_SUCCESS;
|
||
|
}
|
||
|
|
||
|
static int carlu_run_random_test(void)
|
||
|
{
|
||
|
struct carlu *carl = NULL;
|
||
|
int err;
|
||
|
|
||
|
err = carlu_init();
|
||
|
if (err)
|
||
|
goto out;
|
||
|
|
||
|
carl = carlusb_probe();
|
||
|
if (IS_ERR_OR_NULL(carl)) {
|
||
|
err = PTR_ERR(carl);
|
||
|
goto out;
|
||
|
}
|
||
|
|
||
|
err = carlu_random_test(carl);
|
||
|
if (err)
|
||
|
goto out_close;
|
||
|
|
||
|
out_close:
|
||
|
carlusb_close(carl);
|
||
|
|
||
|
out:
|
||
|
carlu_exit();
|
||
|
return err ? EXIT_FAILURE : EXIT_SUCCESS;
|
||
|
}
|
||
|
|
||
|
static int carlu_run_loop_test(void)
|
||
|
{
|
||
|
struct carlu *carl;
|
||
|
int err;
|
||
|
|
||
|
err = carlu_init();
|
||
|
if (err)
|
||
|
return EXIT_FAILURE;
|
||
|
|
||
|
carl = carlusb_probe();
|
||
|
if (IS_ERR_OR_NULL(carl)) {
|
||
|
err = PTR_ERR(carl);
|
||
|
goto out;
|
||
|
}
|
||
|
|
||
|
carlu_cmd_write_mem(carl, AR9170_MAC_REG_BCN_PERIOD, 0xFFFFFFFF);
|
||
|
carlu_cmd_write_mem(carl, AR9170_MAC_REG_PRETBTT, 0xFFFFFFFF);
|
||
|
|
||
|
/* different payload test */
|
||
|
carlu_loopback_test(carl, 9000, 1000, 1566, 1566);
|
||
|
carlusb_close(carl);
|
||
|
|
||
|
out:
|
||
|
return err ? EXIT_FAILURE : EXIT_SUCCESS;
|
||
|
}
|
||
|
|
||
|
static int carlu_probe_all(void)
|
||
|
{
|
||
|
struct carlu *carl[32] = { 0 };
|
||
|
unsigned int devs;
|
||
|
int ret;
|
||
|
|
||
|
ret = carlu_init();
|
||
|
if (ret)
|
||
|
return EXIT_FAILURE;
|
||
|
|
||
|
for (devs = 0; devs < ARRAY_SIZE(carl); devs++) {
|
||
|
carl[devs] = carlusb_probe();
|
||
|
if (IS_ERR_OR_NULL(carl[devs]))
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
info("Found %d devices\n", devs);
|
||
|
|
||
|
for (; devs > 0; devs--)
|
||
|
carlusb_close(carl[devs - 1]);
|
||
|
|
||
|
carlu_exit();
|
||
|
return EXIT_SUCCESS;
|
||
|
}
|
||
|
|
||
|
struct menu_struct {
|
||
|
char option;
|
||
|
unsigned int parameters;
|
||
|
int (*function)(void);
|
||
|
char help_text[80];
|
||
|
};
|
||
|
|
||
|
#define MENU_ITEM(op, func, helpme) \
|
||
|
{ \
|
||
|
.option = op, \
|
||
|
.parameters = 0, \
|
||
|
.function = func, \
|
||
|
.help_text = helpme, \
|
||
|
}
|
||
|
|
||
|
static int show_help(void);
|
||
|
|
||
|
static const struct menu_struct menu[] = {
|
||
|
[0] = MENU_ITEM('h', show_help, "shows this useless help message text."), /* keep this entry at 0! */
|
||
|
MENU_ITEM('e', carlu_dump_eeprom, "hexdumps eeprom content to stdout."),
|
||
|
MENU_ITEM('l', carlusb_print_known_devices, "list of all known ar9170 usb devices."),
|
||
|
MENU_ITEM('p', carlu_probe_all, "probe all possible devices."),
|
||
|
MENU_ITEM('t', carlu_run_loop_test, "run tx/rx test."),
|
||
|
MENU_ITEM('g', carlu_run_gpio_test, "flash the leds."),
|
||
|
MENU_ITEM('r', carlu_run_random_test, "get random numbers."),
|
||
|
};
|
||
|
|
||
|
static int show_help(void)
|
||
|
{
|
||
|
unsigned int i;
|
||
|
char parameters[ARRAY_SIZE(menu) + 1];
|
||
|
|
||
|
for (i = 0; i < ARRAY_SIZE(menu); i++)
|
||
|
parameters[i] = menu[i].option;
|
||
|
|
||
|
parameters[ARRAY_SIZE(menu)] = '\0';
|
||
|
|
||
|
info("usage: ar9170user -[%s]\n", parameters);
|
||
|
|
||
|
for (i = 0; i < ARRAY_SIZE(menu); i++)
|
||
|
info("\t-%c\t%s\n", menu[i].option, menu[i].help_text);
|
||
|
|
||
|
return EXIT_FAILURE;
|
||
|
}
|
||
|
|
||
|
static int select_menu_item(const char arg)
|
||
|
{
|
||
|
unsigned int i;
|
||
|
|
||
|
for (i = ARRAY_SIZE(menu) - 1; i != 0; i--) {
|
||
|
if (arg == menu[i].option)
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
return menu[i].function();
|
||
|
}
|
||
|
|
||
|
int main(int argc, char *argv[])
|
||
|
{
|
||
|
init_debug();
|
||
|
|
||
|
if (argc != 2 || strlen(argv[1]) != 2 || argv[1][0] != '-')
|
||
|
return show_help();
|
||
|
|
||
|
return select_menu_item(argv[1][1]);
|
||
|
}
|