rockbox/apps/plugins/test_disk.c

326 lines
8.2 KiB
C

/***************************************************************************
* __________ __ ___.
* Open \______ \ ____ ____ | | _\_ |__ _______ ___
* Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
* Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
* Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
* \/ \/ \/ \/ \/
* $Id$
*
* Copyright (C) 2007 Jens Arnold
*
* All files in this archive are subject to the GNU General Public License.
* See the file COPYING in the source tree root for full license agreement.
*
* This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
* KIND, either express or implied.
*
****************************************************************************/
#include "plugin.h"
PLUGIN_HEADER
#define TEST_FILE "/test_disk.tmp"
#define FRND_SEED 0x78C3 /* arbirary */
#ifdef HAVE_MMC
#define TEST_SIZE (20*1024*1024)
#else
#define TEST_SIZE (300*1024*1024)
#endif
static struct plugin_api* rb;
static unsigned char* audiobuf;
static int audiobufsize;
static unsigned short frnd_buffer;
static int line = 0;
static int max_line = 0;
static void mem_fill_frnd(unsigned char *addr, int len)
{
unsigned char *end = addr + len;
unsigned random = frnd_buffer;
while (addr < end)
{
random = 75 * random + 74;
*addr++ = random >> 8;
}
frnd_buffer = random;
}
static bool mem_cmp_frnd(unsigned char *addr, int len)
{
unsigned char *end = addr + len;
unsigned random = frnd_buffer;
while (addr < end)
{
random = 75 * random + 74;
if (*addr++ != ((random >> 8) & 0xff))
return false;
}
frnd_buffer = random;
return true;
}
static void log_init(void)
{
int h;
rb->lcd_getstringsize("A", NULL, &h);
max_line = LCD_HEIGHT / h;
line = 0;
rb->lcd_clear_display();
rb->lcd_update();
}
static void log_lcd(char *text, bool advance)
{
rb->lcd_puts(0, line, text);
rb->lcd_update();
if (advance)
if (++line >= max_line)
line = 0;
}
static bool test_fs(void)
{
unsigned char text_buf[32];
unsigned char *buf_start;
int align, buf_len;
int total, current;
int fd;
align = (-(int)audiobuf) & 3;
buf_start = audiobuf + align;
buf_len = (audiobufsize - align - 4) & ~3;
log_init();
rb->snprintf(text_buf, sizeof text_buf, "FS stress test: %dKB", (TEST_SIZE>>10));
log_lcd(text_buf, true);
fd = rb->creat(TEST_FILE);
if (fd < 0)
{
rb->splash(0, true, "Couldn't create testfile.");
goto error;
}
frnd_buffer = FRND_SEED;
total = TEST_SIZE;
while (total > 0)
{
current = rb->rand() % buf_len;
current = MIN(current, total);
align = rb->rand() & 3;
rb->snprintf(text_buf, sizeof text_buf, "Wrt %dKB, %dKB left",
current >> 10, total >> 10);
log_lcd(text_buf, false);
mem_fill_frnd(buf_start + align, current);
if (current != rb->write(fd, buf_start + align, current))
{
rb->splash(0, true, "Write error.");
rb->close(fd);
goto error;
}
total -= current;
}
rb->close(fd);
fd = rb->open(TEST_FILE, O_RDONLY);
if (fd < 0)
{
rb->splash(0, true, "Couldn't open testfile.");
goto error;
}
frnd_buffer = FRND_SEED;
total = TEST_SIZE;
while (total > 0)
{
current = rb->rand() % buf_len;
current = MIN(current, total);
align = rb->rand() & 3;
rb->snprintf(text_buf, sizeof text_buf, "Cmp %dKB, %dKB left",
current >> 10, total >> 10);
log_lcd(text_buf, false);
if (current != rb->read(fd, buf_start + align, current))
{
rb->splash(0, true, "Read error.");
rb->close(fd);
goto error;
}
if (!mem_cmp_frnd(buf_start + align, current))
{
log_lcd(text_buf, true);
log_lcd("Compare error.", true);
rb->close(fd);
goto error;
}
total -= current;
}
rb->close(fd);
log_lcd(text_buf, true);
log_lcd("Test passed.", true);
error:
rb->remove(TEST_FILE);
rb->button_clear_queue();
rb->button_get(true);
return false;
}
static bool test_speed(void)
{
unsigned char text_buf[32];
unsigned char *buf_start;
long time;
int buf_len;
int fd;
buf_len = (-(int)audiobuf) & 3;
buf_start = audiobuf + buf_len;
buf_len = (audiobufsize - buf_len) & ~3;
rb->memset(buf_start, 'T', buf_len);
buf_len -= 2;
log_init();
log_lcd("Disk speed test", true);
fd = rb->creat(TEST_FILE);
if (fd < 0)
{
rb->splash(0, true, "Couldn't create testfile.");
goto error;
}
time = *rb->current_tick;
if (buf_len != rb->write(fd, buf_start, buf_len))
{
rb->splash(0, true, "Write error.");
rb->close(fd);
goto error;
}
time = *rb->current_tick - time;
rb->snprintf(text_buf, sizeof text_buf, "Create: %d KByte/s",
(25 * buf_len / time) >> 8);
log_lcd(text_buf, true);
rb->close(fd);
fd = rb->open(TEST_FILE, O_WRONLY);
if (fd < 0)
{
rb->splash(0, true, "Couldn't open testfile.");
goto error;
}
time = *rb->current_tick;
if (buf_len != rb->write(fd, buf_start, buf_len))
{
rb->splash(0, true, "Write error.");
rb->close(fd);
goto error;
}
time = *rb->current_tick - time;
rb->snprintf(text_buf, sizeof text_buf, "Write A: %d KByte/s",
(25 * buf_len / time) >> 8);
log_lcd(text_buf, true);
rb->close(fd);
fd = rb->open(TEST_FILE, O_WRONLY);
if (fd < 0)
{
rb->splash(0, true, "Couldn't open testfile.");
goto error;
}
time = *rb->current_tick;
if (buf_len != rb->write(fd, buf_start + 1, buf_len))
{
rb->splash(0, true, "Write error.");
rb->close(fd);
goto error;
}
time = *rb->current_tick - time;
rb->snprintf(text_buf, sizeof text_buf, "Write U: %d KByte/s",
(25 * buf_len / time) >> 8);
log_lcd(text_buf, true);
rb->close(fd);
fd = rb->open(TEST_FILE, O_RDONLY);
if (fd < 0)
{
rb->splash(0, true, "Couldn't open testfile.");
goto error;
}
time = *rb->current_tick;
if (buf_len != rb->read(fd, buf_start, buf_len))
{
rb->splash(0, true, "Read error.");
rb->close(fd);
goto error;
}
time = *rb->current_tick - time;
rb->snprintf(text_buf, sizeof text_buf, "Read A: %d KByte/s",
(25 * buf_len / time) >> 8);
log_lcd(text_buf, true);
rb->close(fd);
fd = rb->open(TEST_FILE, O_RDONLY);
if (fd < 0)
{
rb->splash(0, true, "Couldn't open testfile.");
goto error;
}
time = *rb->current_tick;
if (buf_len != rb->read(fd, buf_start + 1, buf_len))
{
rb->splash(0, true, "Read error.");
rb->close(fd);
goto error;
}
time = *rb->current_tick - time;
rb->snprintf(text_buf, sizeof text_buf, "Read U: %d KByte/s",
(25 * buf_len / time) >> 8);
log_lcd(text_buf, true);
rb->close(fd);
error:
rb->remove(TEST_FILE);
rb->button_clear_queue();
rb->button_get(true);
return false;
}
/* this is the plugin entry point */
enum plugin_status plugin_start(struct plugin_api* api, void* parameter)
{
static const struct menu_item items[] = {
{ "Disk speed", test_speed },
{ "FS stress test", test_fs },
};
int m;
(void)parameter;
rb = api;
audiobuf = rb->plugin_get_audio_buffer(&audiobufsize);
rb->srand(*rb->current_tick);
if (rb->global_settings->backlight_timeout > 0)
rb->backlight_set_timeout(1); /* keep the light on */
m = rb->menu_init(items, sizeof(items) / sizeof(*items), NULL,
NULL, NULL, NULL);
rb->menu_run(m);
rb->menu_exit(m);
/* restore normal backlight setting */
rb->backlight_set_timeout(rb->global_settings->backlight_timeout);
return PLUGIN_OK;
}