forked from len0rd/rockbox
		
	git-svn-id: svn://svn.rockbox.org/rockbox/trunk@12008 a1c6a512-1295-4272-9138-f99709370657
		
			
				
	
	
		
			722 lines
		
	
	
	
		
			23 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			722 lines
		
	
	
	
		
			23 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /***************************************************************************
 | |
| *             __________               __   ___.
 | |
| *   Open      \______   \ ____   ____ |  | _\_ |__   _______  ___
 | |
| *   Source     |       _//  _ \_/ ___\|  |/ /| __ \ /  _ \  \/  /
 | |
| *   Jukebox    |    |   (  <_> )  \___|    < | \_\ (  <_> > <  <
 | |
| *   Firmware   |____|_  /\____/ \___  >__|_ \|___  /\____/__/\_ \
 | |
| *                     \/            \/     \/    \/            \/
 | |
| * $Id$
 | |
| *
 | |
| * Oscilloscope, with many different modes of operation.
 | |
| *
 | |
| * Copyright (C) 2004-2006 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"
 | |
| 
 | |
| #ifdef HAVE_LCD_BITMAP
 | |
| #include "xlcd.h"
 | |
| #include "configfile.h"
 | |
| 
 | |
| PLUGIN_HEADER
 | |
| 
 | |
| /* variable button definitions */
 | |
| #if CONFIG_KEYPAD == RECORDER_PAD
 | |
| #define OSCILLOSCOPE_QUIT         BUTTON_OFF
 | |
| #define OSCILLOSCOPE_DRAWMODE     BUTTON_F1
 | |
| #define OSCILLOSCOPE_ADVMODE      BUTTON_F2
 | |
| #define OSCILLOSCOPE_ORIENTATION  BUTTON_F3
 | |
| #define OSCILLOSCOPE_PAUSE        BUTTON_PLAY
 | |
| #define OSCILLOSCOPE_SPEED_UP     BUTTON_RIGHT
 | |
| #define OSCILLOSCOPE_SPEED_DOWN   BUTTON_LEFT
 | |
| #define OSCILLOSCOPE_VOL_UP       BUTTON_UP
 | |
| #define OSCILLOSCOPE_VOL_DOWN     BUTTON_DOWN
 | |
| 
 | |
| #elif CONFIG_KEYPAD == ARCHOS_AV300_PAD
 | |
| #define OSCILLOSCOPE_QUIT         BUTTON_OFF
 | |
| #define OSCILLOSCOPE_DRAWMODE     BUTTON_F1
 | |
| #define OSCILLOSCOPE_ADVMODE      BUTTON_F2
 | |
| #define OSCILLOSCOPE_ORIENTATION  BUTTON_F3
 | |
| #define OSCILLOSCOPE_PAUSE        BUTTON_SELECT
 | |
| #define OSCILLOSCOPE_SPEED_UP     BUTTON_RIGHT
 | |
| #define OSCILLOSCOPE_SPEED_DOWN   BUTTON_LEFT
 | |
| #define OSCILLOSCOPE_VOL_UP       BUTTON_UP
 | |
| #define OSCILLOSCOPE_VOL_DOWN     BUTTON_DOWN
 | |
| 
 | |
| #elif CONFIG_KEYPAD == ONDIO_PAD
 | |
| #define OSCILLOSCOPE_QUIT         BUTTON_OFF
 | |
| #define OSCILLOSCOPE_DRAWMODE_PRE BUTTON_MENU
 | |
| #define OSCILLOSCOPE_DRAWMODE     (BUTTON_MENU | BUTTON_REL)
 | |
| #define OSCILLOSCOPE_ADVMODE      (BUTTON_MENU | BUTTON_RIGHT)
 | |
| #define OSCILLOSCOPE_ORIENTATION  (BUTTON_MENU | BUTTON_LEFT)
 | |
| #define OSCILLOSCOPE_PAUSE        (BUTTON_MENU | BUTTON_OFF)
 | |
| #define OSCILLOSCOPE_SPEED_UP     BUTTON_RIGHT
 | |
| #define OSCILLOSCOPE_SPEED_DOWN   BUTTON_LEFT
 | |
| #define OSCILLOSCOPE_VOL_UP       BUTTON_UP
 | |
| #define OSCILLOSCOPE_VOL_DOWN     BUTTON_DOWN
 | |
| 
 | |
| #elif (CONFIG_KEYPAD == IRIVER_H100_PAD) || (CONFIG_KEYPAD == IRIVER_H300_PAD)
 | |
| #define OSCILLOSCOPE_QUIT         BUTTON_OFF
 | |
| #define OSCILLOSCOPE_DRAWMODE     BUTTON_SELECT
 | |
| #define OSCILLOSCOPE_ADVMODE      BUTTON_MODE
 | |
| #define OSCILLOSCOPE_ORIENTATION  BUTTON_REC
 | |
| #define OSCILLOSCOPE_PAUSE        BUTTON_ON
 | |
| #define OSCILLOSCOPE_SPEED_UP     BUTTON_RIGHT
 | |
| #define OSCILLOSCOPE_SPEED_DOWN   BUTTON_LEFT
 | |
| #define OSCILLOSCOPE_VOL_UP       BUTTON_UP
 | |
| #define OSCILLOSCOPE_VOL_DOWN     BUTTON_DOWN
 | |
| 
 | |
| #define OSCILLOSCOPE_RC_QUIT      BUTTON_RC_STOP
 | |
| #elif (CONFIG_KEYPAD == IPOD_3G_PAD) || (CONFIG_KEYPAD == IPOD_4G_PAD)
 | |
| #define OSCILLOSCOPE_QUIT         (BUTTON_SELECT | BUTTON_MENU)
 | |
| #define OSCILLOSCOPE_DRAWMODE     (BUTTON_SELECT | BUTTON_PLAY)
 | |
| #define OSCILLOSCOPE_ADVMODE      (BUTTON_SELECT | BUTTON_RIGHT)
 | |
| #define OSCILLOSCOPE_ORIENTATION  (BUTTON_SELECT | BUTTON_LEFT)
 | |
| #define OSCILLOSCOPE_PAUSE        BUTTON_PLAY
 | |
| #define OSCILLOSCOPE_SPEED_UP     BUTTON_RIGHT
 | |
| #define OSCILLOSCOPE_SPEED_DOWN   BUTTON_LEFT
 | |
| #define OSCILLOSCOPE_VOL_UP       BUTTON_SCROLL_FWD
 | |
| #define OSCILLOSCOPE_VOL_DOWN     BUTTON_SCROLL_BACK
 | |
| 
 | |
| #elif (CONFIG_KEYPAD == GIGABEAT_PAD)
 | |
| #define OSCILLOSCOPE_QUIT         BUTTON_POWER
 | |
| #define OSCILLOSCOPE_DRAWMODE     BUTTON_SELECT
 | |
| #define OSCILLOSCOPE_ADVMODE      BUTTON_DOWN
 | |
| #define OSCILLOSCOPE_ORIENTATION  BUTTON_UP
 | |
| #define OSCILLOSCOPE_PAUSE        BUTTON_A
 | |
| #define OSCILLOSCOPE_SPEED_UP     BUTTON_RIGHT
 | |
| #define OSCILLOSCOPE_SPEED_DOWN   BUTTON_LEFT
 | |
| #define OSCILLOSCOPE_VOL_UP       BUTTON_VOL_UP
 | |
| #define OSCILLOSCOPE_VOL_DOWN     BUTTON_VOL_DOWN
 | |
| 
 | |
| #elif (CONFIG_KEYPAD == SANSA_E200_PAD)
 | |
| #define OSCILLOSCOPE_QUIT         BUTTON_POWER
 | |
| #define OSCILLOSCOPE_DRAWMODE     BUTTON_SELECT
 | |
| #define OSCILLOSCOPE_ADVMODE      BUTTON_DOWN
 | |
| #define OSCILLOSCOPE_ORIENTATION  BUTTON_UP
 | |
| #define OSCILLOSCOPE_PAUSE        BUTTON_REC
 | |
| #define OSCILLOSCOPE_SPEED_UP     BUTTON_RIGHT
 | |
| #define OSCILLOSCOPE_SPEED_DOWN   BUTTON_LEFT
 | |
| #define OSCILLOSCOPE_VOL_UP       BUTTON_SCROLL_UP
 | |
| #define OSCILLOSCOPE_VOL_DOWN     BUTTON_SCROLL_DOWN
 | |
| 
 | |
| #elif CONFIG_KEYPAD == IAUDIO_X5_PAD
 | |
| #define OSCILLOSCOPE_QUIT         BUTTON_POWER
 | |
| #define OSCILLOSCOPE_DRAWMODE_PRE BUTTON_SELECT
 | |
| #define OSCILLOSCOPE_DRAWMODE     (BUTTON_SELECT | BUTTON_REL)
 | |
| #define OSCILLOSCOPE_ADVMODE      BUTTON_REC
 | |
| #define OSCILLOSCOPE_ORIENTATION  (BUTTON_SELECT | BUTTON_REPEAT)
 | |
| #define OSCILLOSCOPE_PAUSE        BUTTON_PLAY
 | |
| #define OSCILLOSCOPE_SPEED_UP     BUTTON_RIGHT
 | |
| #define OSCILLOSCOPE_SPEED_DOWN   BUTTON_LEFT
 | |
| #define OSCILLOSCOPE_VOL_UP       BUTTON_UP
 | |
| #define OSCILLOSCOPE_VOL_DOWN     BUTTON_DOWN
 | |
| 
 | |
| #elif CONFIG_KEYPAD == IRIVER_H10_PAD
 | |
| #define OSCILLOSCOPE_QUIT         BUTTON_POWER
 | |
| #define OSCILLOSCOPE_DRAWMODE_PRE BUTTON_REW
 | |
| #define OSCILLOSCOPE_DRAWMODE     (BUTTON_REW | BUTTON_REL)
 | |
| #define OSCILLOSCOPE_ADVMODE      BUTTON_FF
 | |
| #define OSCILLOSCOPE_ORIENTATION  (BUTTON_REW | BUTTON_REPEAT)
 | |
| #define OSCILLOSCOPE_PAUSE        BUTTON_PLAY
 | |
| #define OSCILLOSCOPE_SPEED_UP     BUTTON_RIGHT
 | |
| #define OSCILLOSCOPE_SPEED_DOWN   BUTTON_LEFT
 | |
| #define OSCILLOSCOPE_VOL_UP       BUTTON_SCROLL_UP
 | |
| #define OSCILLOSCOPE_VOL_DOWN     BUTTON_SCROLL_DOWN
 | |
| 
 | |
| #endif
 | |
| 
 | |
| /* colours */
 | |
| #if LCD_DEPTH > 1
 | |
| #ifdef HAVE_LCD_COLOR
 | |
| #define BACKG_COLOR  LCD_BLACK
 | |
| #define GRAPH_COLOR  LCD_RGBPACK(128, 255, 0)
 | |
| #define CURSOR_COLOR LCD_RGBPACK(255, 0, 0)
 | |
| #else
 | |
| #define BACKG_COLOR  LCD_WHITE
 | |
| #define GRAPH_COLOR  LCD_BLACK
 | |
| #define CURSOR_COLOR LCD_DARKGRAY
 | |
| #endif
 | |
| #endif
 | |
| 
 | |
| enum { DRAW_FILLED, DRAW_LINE, DRAW_PIXEL, MAX_DRAW };
 | |
| enum { ADV_SCROLL, ADV_WRAP, MAX_ADV };
 | |
| enum { OSC_HORIZ, OSC_VERT, MAX_OSC };
 | |
| 
 | |
| #define CFGFILE_VERSION 0     /* Current config file version */
 | |
| #define CFGFILE_MINVERSION 0  /* Minimum config file version to accept */
 | |
| 
 | |
| 
 | |
| #define MAX_PEAK 0x8000
 | |
| 
 | |
| #if defined(SIMULATOR) && (CONFIG_CODEC != SWCODEC)
 | |
| #define mas_codec_readreg(x) rand()%MAX_PEAK
 | |
| #endif
 | |
| 
 | |
| /* global variables */
 | |
| 
 | |
| struct plugin_api* rb;     /* global api struct pointer */
 | |
| 
 | |
| /* settings */
 | |
| struct osc_config {
 | |
|     int delay;     /* in ticks */
 | |
|     int draw;
 | |
|     int advance;
 | |
|     int orientation;
 | |
| };
 | |
| 
 | |
| struct osc_config osc_disk = { 2, DRAW_FILLED, ADV_SCROLL, OSC_HORIZ };
 | |
| struct osc_config osc;       /* running config */
 | |
| 
 | |
| static const char cfg_filename[] =  "oscilloscope.cfg";
 | |
| static char *draw_str[3]        = { "filled", "line", "pixel" };
 | |
| static char *advance_str[2]     = { "scroll", "wrap" };
 | |
| static char *orientation_str[2] = { "horizontal", "vertical" };
 | |
| 
 | |
| struct configdata disk_config[] = {
 | |
|    { TYPE_INT,  1, 99, &osc_disk.delay, "delay", NULL, NULL },
 | |
|    { TYPE_ENUM, 0, MAX_DRAW, &osc_disk.draw, "draw", draw_str, NULL },
 | |
|    { TYPE_ENUM, 0, MAX_ADV, &osc_disk.advance, "advance", advance_str, NULL },
 | |
|    { TYPE_ENUM, 0, MAX_OSC, &osc_disk.orientation, "orientation", orientation_str, NULL }
 | |
| };
 | |
| 
 | |
| 
 | |
| long last_tick = 0;  /* time of last drawing */
 | |
| int last_pos = 0;    /* last x or y drawing position. Reset for aspect switch. */
 | |
| int last_left;       /* last channel values */
 | |
| int last_right;
 | |
| 
 | |
| unsigned char message[16]; /* message to display */
 | |
| bool displaymsg = false;
 | |
| int font_height = 8;
 | |
| 
 | |
| /* implementation */
 | |
| 
 | |
| void anim_horizontal(int cur_left, int cur_right)
 | |
| {
 | |
|     int cur_x, x;
 | |
|     int left, right, dl, dr;
 | |
|     long cur_tick = *rb->current_tick;
 | |
|     long d = (cur_tick - last_tick) / osc.delay;
 | |
|     bool full_update = false;
 | |
| 
 | |
|     if (d == 0)  /* too early, bail out */
 | |
|         return;
 | |
| 
 | |
|     last_tick = cur_tick;
 | |
| 
 | |
|     if (d > HZ) /* first call or too much delay, (re)start */
 | |
|     {
 | |
|         last_left = cur_left;
 | |
|         last_right = cur_right;
 | |
|         return;
 | |
|     }
 | |
|     cur_x = last_pos + d;
 | |
| 
 | |
|     if (cur_x >= LCD_WIDTH)
 | |
|     {
 | |
|         if (osc.advance == ADV_SCROLL)
 | |
|         {
 | |
|             int shift = cur_x - (LCD_WIDTH-1);
 | |
|             xlcd_scroll_left(shift);
 | |
|             full_update = true;
 | |
|             cur_x -= shift;
 | |
|             last_pos -= shift;
 | |
|         }
 | |
|         else
 | |
|         {
 | |
|             cur_x -= LCD_WIDTH;
 | |
|         }
 | |
|     }
 | |
|     rb->lcd_set_drawmode(DRMODE_SOLID|DRMODE_INVERSEVID);
 | |
| 
 | |
|     if (cur_x > last_pos)
 | |
|     {
 | |
|         rb->lcd_fillrect(last_pos + 1, 0, d, LCD_HEIGHT);
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|         rb->lcd_fillrect(last_pos + 1, 0, LCD_WIDTH - last_pos, LCD_HEIGHT);
 | |
|         rb->lcd_fillrect(0, 0, cur_x + 1, LCD_HEIGHT);
 | |
|     }
 | |
|     rb->lcd_set_drawmode(DRMODE_SOLID);
 | |
| 
 | |
|     switch (osc.draw)
 | |
|     {
 | |
|         case DRAW_FILLED:
 | |
|             left = last_left;
 | |
|             right = last_right;
 | |
|             dl = (cur_left  - left)  / d;
 | |
|             dr = (cur_right - right) / d;
 | |
| 
 | |
|             for (x = last_pos + 1; d > 0; x++, d--)
 | |
|             {
 | |
|                 if (x == LCD_WIDTH)
 | |
|                     x = 0;
 | |
| 
 | |
|                 left  += dl;
 | |
|                 right += dr;
 | |
| 
 | |
|                 rb->lcd_vline(x, LCD_HEIGHT/2-1,
 | |
|                               LCD_HEIGHT/2-1 - (((LCD_HEIGHT-2) * left) >> 16));
 | |
|                 rb->lcd_vline(x, LCD_HEIGHT/2+1,
 | |
|                               LCD_HEIGHT/2+1 + (((LCD_HEIGHT-2) * right) >> 16));
 | |
|             }
 | |
|             break;
 | |
| 
 | |
|         case DRAW_LINE:
 | |
|             if (cur_x > last_pos)
 | |
|             {
 | |
|                 rb->lcd_drawline(
 | |
|                     last_pos, LCD_HEIGHT/2-1 - (((LCD_HEIGHT-2) * last_left) >> 16),
 | |
|                     cur_x, LCD_HEIGHT/2-1 - (((LCD_HEIGHT-2) * cur_left) >> 16)
 | |
|                 );
 | |
|                 rb->lcd_drawline(
 | |
|                     last_pos, LCD_HEIGHT/2+1 + (((LCD_HEIGHT-2) * last_right) >> 16),
 | |
|                     cur_x, LCD_HEIGHT/2+1 + (((LCD_HEIGHT-2) * cur_right) >> 16)
 | |
|                 );
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 left  = last_left 
 | |
|                       + (LCD_WIDTH - last_pos) * (last_left - cur_left) / d;
 | |
|                 right = last_right 
 | |
|                       + (LCD_WIDTH - last_pos) * (last_right - cur_right) / d;
 | |
| 
 | |
|                 rb->lcd_drawline(
 | |
|                     last_pos, LCD_HEIGHT/2-1 - (((LCD_HEIGHT-2) * last_left) >> 16),
 | |
|                     LCD_WIDTH, LCD_HEIGHT/2-1 - (((LCD_HEIGHT-2) * left) >> 16)
 | |
|                 );
 | |
|                 rb->lcd_drawline(
 | |
|                     last_pos, LCD_HEIGHT/2+1 + (((LCD_HEIGHT-2) * last_right) >> 16),
 | |
|                     LCD_WIDTH, LCD_HEIGHT/2+1 + (((LCD_HEIGHT-2) * right) >> 16)
 | |
|                 );
 | |
|                 if (cur_x > 0)
 | |
|                 {
 | |
|                     rb->lcd_drawline(
 | |
|                         0, LCD_HEIGHT/2-1 - (((LCD_HEIGHT-2) * left) >> 16),
 | |
|                         cur_x, LCD_HEIGHT/2-1 - (((LCD_HEIGHT-2) * cur_left) >> 16)
 | |
|                     );
 | |
|                     rb->lcd_drawline(
 | |
|                         0, LCD_HEIGHT/2+1 + (((LCD_HEIGHT-2) * right) >> 16),
 | |
|                         cur_x, LCD_HEIGHT/2+1 + (((LCD_HEIGHT-2) * cur_right) >> 16)
 | |
|                     );
 | |
|                 }
 | |
|             }
 | |
|             break;
 | |
|             
 | |
|         case DRAW_PIXEL:
 | |
|             left = last_left;
 | |
|             right = last_right;
 | |
|             dl = (cur_left - left) / d;
 | |
|             dr = (cur_right - right) / d;
 | |
| 
 | |
|             for (x = last_pos + 1; d > 0; x++, d--)
 | |
|             {
 | |
|                 if (x == LCD_WIDTH)
 | |
|                     x = 0;
 | |
| 
 | |
|                 left  += dl;
 | |
|                 right += dr;
 | |
| 
 | |
|                 rb->lcd_drawpixel(x, LCD_HEIGHT/2-1 - (((LCD_HEIGHT-2) * left) >> 16));
 | |
|                 rb->lcd_drawpixel(x, LCD_HEIGHT/2+1 + (((LCD_HEIGHT-2) * right) >> 16));
 | |
|             }
 | |
|             break;
 | |
| 
 | |
|     }
 | |
|     last_left  = cur_left;
 | |
|     last_right = cur_right;
 | |
|     
 | |
|     if (displaymsg)
 | |
|     {
 | |
|         int width;
 | |
|         
 | |
|         rb->lcd_getstringsize(message, &width, NULL);
 | |
|         last_pos -= width - 1;
 | |
|         rb->lcd_putsxy(last_pos, 0, message);
 | |
|         displaymsg = false;
 | |
| 
 | |
|         if (last_pos < 0)
 | |
|             last_pos = 0;
 | |
|     }
 | |
| 
 | |
|     if (full_update)
 | |
|     {
 | |
|         rb->lcd_update();
 | |
|     }
 | |
|     else
 | |
|     {
 | |
| #if LCD_DEPTH > 1                 /* cursor bar */    
 | |
|         rb->lcd_set_foreground(CURSOR_COLOR);
 | |
|         rb->lcd_vline(cur_x + 1, 0, LCD_HEIGHT-1); 
 | |
|         rb->lcd_set_foreground(GRAPH_COLOR);
 | |
| #else
 | |
|         rb->lcd_set_drawmode(DRMODE_COMPLEMENT);
 | |
|         rb->lcd_vline(cur_x + 1, 0, LCD_HEIGHT-1);
 | |
|         rb->lcd_set_drawmode(DRMODE_SOLID);
 | |
| #endif
 | |
| 
 | |
|         if (cur_x > last_pos)
 | |
|         {
 | |
|             rb->lcd_update_rect(last_pos, 0, cur_x - last_pos + 2, LCD_HEIGHT);
 | |
|         }
 | |
|         else
 | |
|         {
 | |
|             rb->lcd_update_rect(last_pos, 0, LCD_WIDTH - last_pos, LCD_HEIGHT);
 | |
|             rb->lcd_update_rect(0, 0, cur_x + 2, LCD_HEIGHT);
 | |
|         }
 | |
|     }
 | |
|     last_pos = cur_x;
 | |
| }
 | |
| 
 | |
| void anim_vertical(int cur_left, int cur_right)
 | |
| {
 | |
|     int cur_y, y;
 | |
|     int left, right, dl, dr;
 | |
|     long cur_tick = *rb->current_tick;
 | |
|     long d = (cur_tick - last_tick) / osc.delay;
 | |
|     bool full_update = false;
 | |
| 
 | |
|     if (d == 0)  /* too early, bail out */
 | |
|         return;
 | |
| 
 | |
|     last_tick = cur_tick;
 | |
| 
 | |
|     if (d > HZ) /* first call or too much delay, (re)start */
 | |
|     {
 | |
|         last_left = cur_left;
 | |
|         last_right = cur_right;
 | |
|         return;
 | |
|     }
 | |
|     cur_y = last_pos + d;
 | |
| 
 | |
|     if (cur_y >= LCD_HEIGHT)
 | |
|     {
 | |
|         if (osc.advance == ADV_SCROLL)
 | |
|         {
 | |
|             int shift = cur_y - (LCD_HEIGHT-1);
 | |
|             xlcd_scroll_up(shift);
 | |
|             full_update = true;
 | |
|             cur_y -= shift;
 | |
|             last_pos -= shift;
 | |
|         }
 | |
|         else 
 | |
|         {
 | |
|             cur_y -= LCD_HEIGHT;
 | |
|         }
 | |
|     }
 | |
|     rb->lcd_set_drawmode(DRMODE_SOLID|DRMODE_INVERSEVID);
 | |
| 
 | |
|     if (cur_y > last_pos)
 | |
|     {
 | |
|         rb->lcd_fillrect(0, last_pos + 1, LCD_WIDTH, d);
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|         rb->lcd_fillrect(0, last_pos + 1, LCD_WIDTH, LCD_HEIGHT - last_pos);
 | |
|         rb->lcd_fillrect(0, 0, LCD_WIDTH, cur_y + 1);
 | |
|     }
 | |
|     rb->lcd_set_drawmode(DRMODE_SOLID);
 | |
| 
 | |
|     switch (osc.draw)
 | |
|     {
 | |
|         case DRAW_FILLED:
 | |
|             left = last_left;
 | |
|             right = last_right;
 | |
|             dl = (cur_left  - left)  / d;
 | |
|             dr = (cur_right - right) / d;
 | |
| 
 | |
|             for (y = last_pos + 1; d > 0; y++, d--)
 | |
|             {
 | |
|                 if (y == LCD_HEIGHT)
 | |
|                     y = 0;
 | |
| 
 | |
|                 left  += dl;
 | |
|                 right += dr;
 | |
| 
 | |
|                 rb->lcd_hline(LCD_WIDTH/2-1,
 | |
|                               LCD_WIDTH/2-1 - (((LCD_WIDTH-2) * left) >> 16), y);
 | |
|                 rb->lcd_hline(LCD_WIDTH/2+1,
 | |
|                               LCD_WIDTH/2+1 + (((LCD_WIDTH-2) * right) >> 16), y);
 | |
|             }
 | |
|             break;
 | |
| 
 | |
|         case DRAW_LINE:
 | |
|             if (cur_y > last_pos)
 | |
|             {
 | |
|                 rb->lcd_drawline(
 | |
|                     LCD_WIDTH/2-1 - (((LCD_WIDTH-2) * last_left) >> 16), last_pos,
 | |
|                     LCD_WIDTH/2-1 - (((LCD_WIDTH-2) * cur_left) >> 16), cur_y
 | |
|                 );
 | |
|                 rb->lcd_drawline(
 | |
|                     LCD_WIDTH/2+1 + (((LCD_WIDTH-2) * last_right) >> 16), last_pos,
 | |
|                     LCD_WIDTH/2+1 + (((LCD_WIDTH-2) * cur_right) >> 16), cur_y
 | |
|                 );
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 left  = last_left 
 | |
|                       + (LCD_HEIGHT - last_pos) * (last_left - cur_left) / d;
 | |
|                 right = last_right
 | |
|                       + (LCD_HEIGHT - last_pos) * (last_right - cur_right) / d;
 | |
| 
 | |
|                 rb->lcd_drawline(
 | |
|                     LCD_WIDTH/2-1 - (((LCD_WIDTH-2) * last_left) >> 16), last_pos,
 | |
|                     LCD_WIDTH/2-1 - (((LCD_WIDTH-2) * left) >> 16), LCD_HEIGHT
 | |
|                 );
 | |
|                 rb->lcd_drawline(
 | |
|                     LCD_WIDTH/2+1 + (((LCD_WIDTH-2) * last_right) >> 16), last_pos,
 | |
|                     LCD_WIDTH/2+1 + (((LCD_WIDTH-2) * right) >> 16), LCD_HEIGHT
 | |
|                 );
 | |
|                 if (cur_y > 0)
 | |
|                 {
 | |
|                     rb->lcd_drawline(
 | |
|                         LCD_WIDTH/2-1 - (((LCD_WIDTH-2) * left) >> 16), 0,
 | |
|                         LCD_WIDTH/2-1 - (((LCD_WIDTH-2) * cur_left) >> 16), cur_y
 | |
|                     );
 | |
|                     rb->lcd_drawline(
 | |
|                         LCD_WIDTH/2+1 + (((LCD_WIDTH-2) * right) >> 16), 0,
 | |
|                         LCD_WIDTH/2+1 + (((LCD_WIDTH-2) * cur_right) >> 16), cur_y
 | |
|                     );
 | |
|                 }
 | |
|             }
 | |
|             break;
 | |
|             
 | |
|         case DRAW_PIXEL:
 | |
|             left = last_left;
 | |
|             right = last_right;
 | |
|             dl = (cur_left - left) / d;
 | |
|             dr = (cur_right - right) / d;
 | |
| 
 | |
|             for (y = last_pos + 1; d > 0; y++, d--)
 | |
|             {
 | |
|                 if (y == LCD_HEIGHT)
 | |
|                     y = 0;
 | |
| 
 | |
|                 left  += dl;
 | |
|                 right += dr;
 | |
| 
 | |
|                 rb->lcd_drawpixel(LCD_WIDTH/2-1 - (((LCD_WIDTH-2) * left) >> 16), y);
 | |
|                 rb->lcd_drawpixel(LCD_WIDTH/2+1 + (((LCD_WIDTH-2) * right) >> 16), y);
 | |
|             }
 | |
|             break;
 | |
| 
 | |
|     }
 | |
|     last_left  = cur_left;
 | |
|     last_right = cur_right;
 | |
|     
 | |
|     if (displaymsg)
 | |
|     {
 | |
|         last_pos -= font_height - 1;
 | |
|         rb->lcd_putsxy(0, last_pos, message);
 | |
|         displaymsg = false;
 | |
| 
 | |
|         if (last_pos < 0)
 | |
|             last_pos = 0;
 | |
|     }
 | |
| 
 | |
|     if (full_update)
 | |
|     {
 | |
|         rb->lcd_update();
 | |
|     }
 | |
|     else
 | |
|     {
 | |
| #if LCD_DEPTH > 1               /* cursor bar */
 | |
|         rb->lcd_set_foreground(CURSOR_COLOR);
 | |
|         rb->lcd_hline(0, LCD_WIDTH-1, cur_y + 1); 
 | |
|         rb->lcd_set_foreground(GRAPH_COLOR);
 | |
| #else
 | |
|         rb->lcd_set_drawmode(DRMODE_COMPLEMENT);
 | |
|         rb->lcd_hline(0, LCD_WIDTH-1, cur_y + 1);
 | |
|         rb->lcd_set_drawmode(DRMODE_SOLID);
 | |
| #endif
 | |
| 
 | |
|         if (cur_y > last_pos)
 | |
|         {
 | |
|             rb->lcd_update_rect(0, last_pos, LCD_WIDTH, cur_y - last_pos + 2);
 | |
|         }
 | |
|         else
 | |
|         {
 | |
|             rb->lcd_update_rect(0, last_pos, LCD_WIDTH, LCD_HEIGHT - last_pos);
 | |
|             rb->lcd_update_rect(0, 0, LCD_WIDTH, cur_y + 2);
 | |
|         }
 | |
|     }
 | |
|     last_pos = cur_y;
 | |
| }
 | |
| 
 | |
| void cleanup(void *parameter)
 | |
| {
 | |
|     (void)parameter;
 | |
| #if LCD_DEPTH > 1
 | |
|     rb->lcd_set_foreground(LCD_DEFAULT_FG);
 | |
|     rb->lcd_set_background(LCD_DEFAULT_BG);
 | |
| #endif
 | |
| #ifdef HAVE_LCD_COLOR
 | |
|     rb->backlight_set_timeout(rb->global_settings->backlight_timeout);
 | |
| #endif
 | |
| }
 | |
| 
 | |
| enum plugin_status plugin_start(struct plugin_api* api, void* parameter)
 | |
| {
 | |
|     int button, vol;
 | |
|     int lastbutton = BUTTON_NONE;
 | |
|     bool exit = false;
 | |
|     bool paused = false;
 | |
|     bool tell_speed;
 | |
| 
 | |
|     (void)parameter;
 | |
|     rb = api;
 | |
| 
 | |
|     xlcd_init(rb);
 | |
|     configfile_init(rb);
 | |
| 
 | |
|     configfile_load(cfg_filename, disk_config,
 | |
|                     sizeof(disk_config) / sizeof(disk_config[0]),
 | |
|                     CFGFILE_MINVERSION);
 | |
|     rb->memcpy(&osc, &osc_disk, sizeof(osc));   /* copy to running config */
 | |
| 
 | |
| #if LCD_DEPTH > 1
 | |
|     rb->lcd_set_foreground(GRAPH_COLOR);
 | |
|     rb->lcd_set_background(BACKG_COLOR);
 | |
|     rb->lcd_set_backdrop(NULL);
 | |
|     rb->lcd_clear_display();
 | |
|     rb->lcd_update();
 | |
| #endif
 | |
| #ifdef HAVE_LCD_COLOR
 | |
|     rb->backlight_set_timeout(1); /* keep the light on */
 | |
| #endif
 | |
|     rb->lcd_getstringsize("A", NULL, &font_height);
 | |
| 
 | |
|     while (!exit)
 | |
|     {
 | |
|         if (!paused)
 | |
|         {
 | |
|             int left, right;
 | |
| 
 | |
|             rb->sleep(MAX(last_tick + osc.delay - *rb->current_tick - 1, 0));
 | |
| 
 | |
| #if (CONFIG_CODEC == MAS3587F) || (CONFIG_CODEC == MAS3539F)
 | |
|             left = rb->mas_codec_readreg(0xC);
 | |
|             right = rb->mas_codec_readreg(0xD);
 | |
| #elif (CONFIG_CODEC == SWCODEC)
 | |
|             rb->pcm_calculate_peaks(&left, &right);
 | |
| #endif
 | |
|             if (osc.orientation == OSC_HORIZ)
 | |
|                 anim_horizontal(left, right);
 | |
|             else
 | |
|                 anim_vertical(left, right);
 | |
|         }
 | |
| 
 | |
|         tell_speed = false;
 | |
|         button = rb->button_get(paused);
 | |
|         switch (button)
 | |
|         {
 | |
| #ifdef OSCILLOSCOPE_RC_QUIT
 | |
|             case OSCILLOSCOPE_RC_QUIT:
 | |
| #endif
 | |
|             case OSCILLOSCOPE_QUIT:
 | |
|                 exit = true;
 | |
|                 break;
 | |
| 
 | |
|             case OSCILLOSCOPE_ADVMODE:
 | |
|                 if (++osc.advance >= MAX_ADV)
 | |
|                     osc.advance = 0;
 | |
|                 break;
 | |
| 
 | |
|             case OSCILLOSCOPE_DRAWMODE:
 | |
| #ifdef OSCILLOSCOPE_DRAWMODE_PRE
 | |
|                 if (lastbutton != OSCILLOSCOPE_DRAWMODE_PRE)
 | |
|                     break;
 | |
| #endif
 | |
|                 if (++osc.draw >= MAX_DRAW)
 | |
|                     osc.draw = 0;
 | |
|                 break;
 | |
|                 
 | |
|             case OSCILLOSCOPE_ORIENTATION:
 | |
|                 if (++osc.orientation >= MAX_OSC)
 | |
|                     osc.orientation = 0;
 | |
|                 last_pos = 0;
 | |
|                 last_tick = 0;
 | |
|                 displaymsg = false;
 | |
|                 rb->lcd_clear_display();
 | |
|                 rb->lcd_update();
 | |
|                 break;
 | |
| 
 | |
|             case OSCILLOSCOPE_PAUSE:
 | |
|                 paused = !paused;
 | |
|                 last_tick = 0;
 | |
|                 break;
 | |
|                 
 | |
|             case OSCILLOSCOPE_SPEED_UP:
 | |
|             case OSCILLOSCOPE_SPEED_UP | BUTTON_REPEAT:
 | |
|                 if (osc.delay > 1)
 | |
|                 {
 | |
|                     osc.delay--;
 | |
|                     tell_speed = true;
 | |
|                 }
 | |
|                 break;
 | |
|                 
 | |
|             case OSCILLOSCOPE_SPEED_DOWN:
 | |
|             case OSCILLOSCOPE_SPEED_DOWN | BUTTON_REPEAT:
 | |
|                 osc.delay++;
 | |
|                 tell_speed = true;
 | |
|                 break;
 | |
| 
 | |
|             case OSCILLOSCOPE_VOL_UP:
 | |
|             case OSCILLOSCOPE_VOL_UP | BUTTON_REPEAT:
 | |
|                 vol = rb->global_settings->volume;
 | |
|                 if (vol < rb->sound_max(SOUND_VOLUME))
 | |
|                 {
 | |
|                     vol++;
 | |
|                     rb->sound_set(SOUND_VOLUME, vol);
 | |
|                     rb->global_settings->volume = vol;
 | |
|                 }
 | |
|                 break;
 | |
| 
 | |
|             case OSCILLOSCOPE_VOL_DOWN:
 | |
|             case OSCILLOSCOPE_VOL_DOWN | BUTTON_REPEAT:
 | |
|                 vol = rb->global_settings->volume;
 | |
|                 if (vol > rb->sound_min(SOUND_VOLUME))
 | |
|                 {
 | |
|                     vol--;
 | |
|                     rb->sound_set(SOUND_VOLUME, vol);
 | |
|                     rb->global_settings->volume = vol;
 | |
|                 }
 | |
|                 break;
 | |
| 
 | |
|             default:
 | |
|                 if (rb->default_event_handler_ex(button, cleanup, NULL)
 | |
|                     == SYS_USB_CONNECTED)
 | |
|                     return PLUGIN_USB_CONNECTED;
 | |
|                 break;
 | |
|         }
 | |
|         if (button != BUTTON_NONE)
 | |
|             lastbutton = button;
 | |
|             
 | |
|         if (tell_speed)
 | |
|         {
 | |
|             rb->snprintf(message, sizeof(message), "%s%d", 
 | |
|                         (osc.orientation == OSC_VERT) ? "Speed: " : "",
 | |
|                         100 / osc.delay);
 | |
|             displaymsg = true;
 | |
|         }
 | |
|     }
 | |
|     cleanup(NULL);
 | |
|     if (rb->memcmp(&osc, &osc_disk, sizeof(osc))) /* save settings if changed */
 | |
|     {
 | |
|         rb->memcpy(&osc_disk, &osc, sizeof(osc));
 | |
|         configfile_save(cfg_filename, disk_config,
 | |
|                         sizeof(disk_config) / sizeof(disk_config[0]),
 | |
|                         CFGFILE_VERSION);
 | |
|     }
 | |
|     return PLUGIN_OK;
 | |
| }
 | |
| #endif /* HAVE_LCD_BITMAP */
 |