ViewVC Help
View File | Revision Log | Show Annotations | Revision Graph | Root Listing
root/cebix/Frodo4/Src/Display_Amiga.h
Revision: 1.3
Committed: 2004-01-12T15:13:20Z (20 years, 8 months ago) by cebix
Content type: text/plain
Branch: MAIN
Changes since 1.2: +1 -1 lines
Log Message:
Happy New Year!

File Contents

# User Rev Content
1 cebix 1.1 /*
2     * Display_Amiga.h - C64 graphics display, emulator window handling,
3     * Amiga specific stuff
4     *
5 cebix 1.3 * Frodo (C) 1994-1997,2002-2004 Christian Bauer
6 cebix 1.1 *
7     * This program is free software; you can redistribute it and/or modify
8     * it under the terms of the GNU General Public License as published by
9     * the Free Software Foundation; either version 2 of the License, or
10     * (at your option) any later version.
11     *
12     * This program is distributed in the hope that it will be useful,
13     * but WITHOUT ANY WARRANTY; without even the implied warranty of
14     * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15     * GNU General Public License for more details.
16     *
17     * You should have received a copy of the GNU General Public License
18     * along with this program; if not, write to the Free Software
19     * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
20     */
21    
22     #include <exec/types.h>
23     #include <exec/memory.h>
24     #include <intuition/intuition.h>
25     #include <libraries/gadtools.h>
26     #include <libraries/asl.h>
27     #include <proto/exec.h>
28     #include <proto/graphics.h>
29     #include <proto/intuition.h>
30     #include <proto/dos.h>
31     #include <proto/gadtools.h>
32     #include <proto/diskfont.h>
33     #include <proto/asl.h>
34    
35     #include "C64.h"
36     #include "SAM.h"
37     #include "Version.h"
38    
39    
40     /*
41     C64 keyboard matrix:
42    
43     Bit 7 6 5 4 3 2 1 0
44     0 CUD F5 F3 F1 F7 CLR RET DEL
45     1 SHL E S Z 4 A W 3
46     2 X T F C 6 D R 5
47     3 V U H B 8 G Y 7
48     4 N O K M 0 J I 9
49     5 , @ : . - L P +
50     6 / ^ = SHR HOM ; * £
51     7 R/S Q C= SPC 2 CTL <- 1
52     */
53    
54    
55     /*
56     Tables for key translation
57     Bit 0..2: row/column in C64 keyboard matrix
58     Bit 3 : implicit shift
59     Bit 5 : joystick emulation (bit 0..4: mask)
60     */
61    
62     const int key_byte[128] = {
63     7, 7, 7, 1, 1, 2, 2, 3,
64     3, 4, 4, 5, 5, 6, -1,0x30,
65     7, 1, 1, 2, 2, 3, 3, 4,
66     4, 5, 5, 6, -1,0x26,0x22,0x2a,
67     1, 1, 2, 2, 3, 3, 4, 4,
68     5, 5, 6, 6, -1,0x24,0x30,0x28,
69     6, 1, 2, 2, 3, 3, 4, 4,
70     5, 5, 6, -1, -1,0x25,0x21,0x29,
71     7, 0, -1, 0, 0, 7, 6, -1,
72     -1, -1, -1, -1,8+0, 0, 0, 8+0,
73     0,8+0, 0,8+0, 0, 8+0, 0, 8+0,
74     -1, -1, 6, 6, -1, -1, -1, -1,
75     1, 6, 1, 7, 7, 7, 7, -1,
76     -1, -1, -1, -1, -1, -1, -1, -1,
77     -1, -1, -1, -1, -1, -1, -1, -1,
78     -1, -1, -1, -1, -1, -1, -1, -1
79     };
80    
81     const int key_bit[128] = {
82     1, 0, 3, 0, 3, 0, 3, 0,
83     3, 0, 3, 0, 3, 0, -1, -1,
84     6, 1, 6, 1, 6, 1, 6, 1,
85     6, 1, 6, 1, -1, -1, -1, -1,
86     2, 5, 2, 5, 2, 5, 2, 5,
87     2, 5, 2, 5, -1, -1, -1, -1,
88     6, 4, 7, 4, 7, 4, 7, 4,
89     7, 4, 7, -1, -1, -1, -1, -1,
90     4, 0, -1, 1, 1, 7, 3, -1,
91     -1, -1, -1, -1, 7, 7, 2, 2,
92     4, 4, 5, 5, 6, 6, 3, 3,
93     -1, -1, 6, 5, -1, -1, -1, -1,
94     7, 4, 7, 2, 5, 5, 5, -1,
95     -1, -1, -1, -1, -1, -1, -1, -1,
96     -1, -1, -1, -1, -1, -1, -1, -1,
97     -1, -1, -1, -1, -1, -1, -1, -1
98     };
99    
100    
101     /*
102     * Menu definitions
103     */
104    
105     const struct NewMenu new_menus[] = {
106     NM_TITLE, "Frodo", NULL, 0, 0, NULL,
107     NM_ITEM, "About Frodo...", NULL, 0, 0, NULL,
108     NM_ITEM, NM_BARLABEL, NULL, 0, 0, NULL,
109     NM_ITEM, "Preferences...", "P", 0, 0, NULL,
110     NM_ITEM, NM_BARLABEL, NULL, 0, 0, NULL,
111     NM_ITEM, "Reset C64", NULL, 0, 0, NULL,
112     NM_ITEM, "Insert next disk", "D", 0, 0, NULL,
113     NM_ITEM, "SAM...", "M", 0, 0, NULL,
114     NM_ITEM, NM_BARLABEL, NULL, 0, 0, NULL,
115     NM_ITEM, "Load snapshot...", "O", 0, 0, NULL,
116     NM_ITEM, "Save snapshot...", "S", 0, 0, NULL,
117     NM_ITEM, NM_BARLABEL, NULL, 0, 0, NULL,
118     NM_ITEM, "Quit Frodo", "Q", 0, 0, NULL,
119     NM_END, NULL, NULL, 0, 0, NULL
120     };
121    
122    
123     /*
124     * Font attributes
125     */
126    
127     const struct TextAttr led_font_attr = {
128     "Helvetica.font", 11, FS_NORMAL, 0
129     };
130    
131     const struct TextAttr speedo_font_attr = {
132     "Courier.font", 11, FS_NORMAL, 0
133     };
134    
135    
136     /*
137     * Display constructor: Create window/screen
138     */
139    
140     C64Display::C64Display(C64 *the_c64) : TheC64(the_c64)
141     {
142     int i;
143    
144     // LEDs off
145     for (i=0; i<4; i++)
146     led_state[i] = old_led_state[i] = LED_OFF;
147    
148     // Allocate chunky buffer to draw into
149     chunky_buf = new UBYTE[DISPLAY_X * DISPLAY_Y];
150    
151     // Open fonts
152     led_font = OpenDiskFont(&led_font_attr);
153     speedo_font = OpenDiskFont(&speedo_font_attr);
154    
155     // Open window on default pubscreen
156     the_window = OpenWindowTags(NULL,
157     WA_Left, 0,
158     WA_Top, 0,
159     WA_InnerWidth, DISPLAY_X,
160     WA_InnerHeight, DISPLAY_Y + 16,
161     WA_Title, (ULONG)"Frodo",
162     WA_ScreenTitle, (ULONG)"Frodo C64 Emulator",
163     WA_IDCMP, IDCMP_CLOSEWINDOW | IDCMP_RAWKEY | IDCMP_MENUPICK | IDCMP_REFRESHWINDOW,
164     WA_DragBar, TRUE,
165     WA_DepthGadget, TRUE,
166     WA_CloseGadget, TRUE,
167     WA_SimpleRefresh, TRUE,
168     WA_Activate, TRUE,
169     WA_NewLookMenus, TRUE,
170     TAG_DONE);
171     the_screen = the_window->WScreen;
172     the_rp = the_window->RPort;
173     xo = the_window->BorderLeft;
174     yo = the_window->BorderTop;
175    
176     // Create menus
177     the_visual_info = GetVisualInfo(the_screen, NULL);
178     the_menus = CreateMenus(new_menus, GTMN_FullMenu, TRUE, TAG_DONE);
179     LayoutMenus(the_menus, the_visual_info, GTMN_NewLookMenus, TRUE, TAG_DONE);
180     SetMenuStrip(the_window, the_menus);
181    
182     // Obtain 16 pens for the C64 colors
183     for (i=0; i<16; i++)
184     pens[i] = ObtainBestPen(the_screen->ViewPort.ColorMap,
185     palette_red[i] * 0x01010101, palette_green[i] * 0x01010101, palette_blue[i] * 0x01010101);
186    
187     // Allocate temporary RastPort for WritePixelArra8()
188     temp_bm = AllocBitMap(DISPLAY_X, 1, 8, 0, NULL);
189     InitRastPort(&temp_rp);
190     temp_rp.BitMap = temp_bm;
191    
192     // Draw LED bar
193     draw_led_bar();
194    
195     // Allocate file requesters
196     open_req = (struct FileRequester *)AllocAslRequestTags(ASL_FileRequest,
197     ASLFR_Window, (ULONG)the_window,
198     ASLFR_SleepWindow, TRUE,
199     ASLFR_TitleText, (ULONG)"Frodo: Load snapshot...",
200     ASLFR_RejectIcons, TRUE,
201     TAG_DONE);
202     save_req = (struct FileRequester *)AllocAslRequestTags(ASL_FileRequest,
203     ASLFR_Window, (ULONG)the_window,
204     ASLFR_SleepWindow, TRUE,
205     ASLFR_TitleText, (ULONG)"Frodo: Save snapshot...",
206     ASLFR_DoSaveMode, TRUE,
207     ASLFR_RejectIcons, TRUE,
208     TAG_DONE);
209     }
210    
211    
212     /*
213     * Display destructor
214     */
215    
216     C64Display::~C64Display()
217     {
218     // Free file requesters
219     if (open_req != NULL)
220     FreeAslRequest(open_req);
221     if (save_req != NULL)
222     FreeAslRequest(save_req);
223    
224     // Free temporary RastPort
225     if (temp_bm != NULL)
226     FreeBitMap(temp_bm);
227    
228     // Free pens
229     for (int i=0; i<16; i++)
230     ReleasePen(the_screen->ViewPort.ColorMap, pens[i]);
231    
232     // Delete menus
233     if (the_menus != NULL) {
234     if (the_window != NULL)
235     ClearMenuStrip(the_window);
236     FreeMenus(the_menus);
237     }
238    
239     // Delete VisualInfo
240     if (the_visual_info != NULL)
241     FreeVisualInfo(the_visual_info);
242    
243     // Close window
244     if (the_window != NULL)
245     CloseWindow(the_window);
246    
247     // Close fonts
248     CloseFont(speedo_font);
249     CloseFont(led_font);
250    
251     // Free chunky buffer
252     delete chunky_buf;
253     }
254    
255    
256     /*
257     * Prefs may have changed
258     */
259    
260     void C64Display::NewPrefs(Prefs *prefs)
261     {
262     }
263    
264    
265     /*
266     * Redraw bitmap
267     */
268    
269     void C64Display::Update(void)
270     {
271     // Update C64 display
272     WritePixelArray8(the_rp, xo, yo, DISPLAY_X + xo - 1, DISPLAY_Y + yo - 1,
273     chunky_buf, &temp_rp);
274    
275     // Update drive LEDs
276     for (int i=0; i<4; i++)
277     if (led_state[i] != old_led_state[i]) {
278     draw_led(i, led_state[i]);
279     old_led_state[i] = led_state[i];
280     }
281     }
282    
283    
284     /*
285     * Draw LED bar at the bottom of the window
286     */
287    
288     void C64Display::draw_led_bar(void)
289     {
290     int i;
291     char str[16];
292    
293     SetAPen(the_rp, pens[15]); // Light gray
294     SetBPen(the_rp, pens[15]); // Light gray
295     RectFill(the_rp, xo, yo+DISPLAY_Y, xo+DISPLAY_X-1, yo+DISPLAY_Y+15);
296    
297     SetAPen(the_rp, pens[1]); // White
298     Move(the_rp, xo, yo+DISPLAY_Y); Draw(the_rp, xo+DISPLAY_X-1, yo+DISPLAY_Y);
299     for (i=0; i<5; i++) {
300     Move(the_rp, xo+DISPLAY_X*i/5, yo+DISPLAY_Y); Draw(the_rp, xo+DISPLAY_X*i/5, yo+DISPLAY_Y+14);
301     }
302     for (i=2; i<6; i++) {
303     Move(the_rp, xo+DISPLAY_X*i/5-23, yo+DISPLAY_Y+11); Draw(the_rp, xo+DISPLAY_X*i/5-9, yo+DISPLAY_Y+11);
304     Move(the_rp, xo+DISPLAY_X*i/5-9, yo+DISPLAY_Y+11); Draw(the_rp, xo+DISPLAY_X*i/5-9, yo+DISPLAY_Y+5);
305     }
306    
307     SetAPen(the_rp, pens[12]); // Medium gray
308     Move(the_rp, xo, yo+DISPLAY_Y+15); Draw(the_rp, xo+DISPLAY_X-1, yo+DISPLAY_Y+15);
309     for (i=1; i<6; i++) {
310     Move(the_rp, xo+DISPLAY_X*i/5-1, yo+DISPLAY_Y+1); Draw(the_rp, xo+DISPLAY_X*i/5-1, yo+DISPLAY_Y+15);
311     }
312     for (i=2; i<6; i++) {
313     Move(the_rp, xo+DISPLAY_X*i/5-24, yo+DISPLAY_Y+11); Draw(the_rp, xo+DISPLAY_X*i/5-24, yo+DISPLAY_Y+4);
314     Move(the_rp, xo+DISPLAY_X*i/5-24, yo+DISPLAY_Y+4); Draw(the_rp, xo+DISPLAY_X*i/5-9, yo+DISPLAY_Y+4);
315     }
316    
317     SetFont(the_rp, led_font);
318     for (i=0; i<4; i++) {
319     sprintf(str, "Drive %d", i+8);
320     SetAPen(the_rp, pens[0]); // Black
321     Move(the_rp, xo+DISPLAY_X*(i+1)/5+8, yo+DISPLAY_Y+11);
322     Text(the_rp, str, strlen(str));
323     draw_led(i, LED_OFF);
324     }
325     }
326    
327    
328     /*
329     * Draw one LED
330     */
331    
332     void C64Display::draw_led(int num, int state)
333     {
334     switch (state) {
335     case LED_OFF:
336     case LED_ERROR_OFF:
337     SetAPen(the_rp, pens[0]); // Black;
338     break;
339     case LED_ON:
340     SetAPen(the_rp, pens[5]); // Green
341     break;
342     case LED_ERROR_ON:
343     SetAPen(the_rp, pens[2]); // Red
344     break;
345     }
346     RectFill(the_rp, xo+DISPLAY_X*(num+2)/5-23, yo+DISPLAY_Y+5, xo+DISPLAY_X*(num+2)/5-10, yo+DISPLAY_Y+10);
347     }
348    
349    
350     /*
351     * Update speedometer
352     */
353    
354     void C64Display::Speedometer(int speed)
355     {
356     static int delay = 0;
357    
358     if (delay >= 20) {
359     char str[16];
360     sprintf(str, "%d%%", speed);
361     SetAPen(the_rp, pens[15]); // Light gray
362     RectFill(the_rp, xo+1, yo+DISPLAY_Y+1, xo+DISPLAY_X/5-2, yo+DISPLAY_Y+14);
363     SetAPen(the_rp, pens[0]); // Black
364     SetFont(the_rp, speedo_font);
365     Move(the_rp, xo+24, yo+DISPLAY_Y+10);
366     Text(the_rp, str, strlen(str));
367     delay = 0;
368     } else
369     delay++;
370     }
371    
372    
373     /*
374     * Return pointer to bitmap data
375     */
376    
377     UBYTE *C64Display::BitmapBase(void)
378     {
379     return chunky_buf;
380     }
381    
382    
383     /*
384     * Return number of bytes per row
385     */
386    
387     int C64Display::BitmapXMod(void)
388     {
389     return DISPLAY_X;
390     }
391    
392    
393     /*
394     * Handle IDCMP messages
395     */
396    
397     void C64Display::PollKeyboard(UBYTE *key_matrix, UBYTE *rev_matrix, UBYTE *joystick)
398     {
399     struct IntuiMessage *msg;
400    
401     // Get and analyze all pending window messages
402     while ((msg = (struct IntuiMessage *)GetMsg(the_window->UserPort)) != NULL) {
403    
404     // Extract data and reply message
405     ULONG iclass = msg->Class;
406     USHORT code = msg->Code;
407     ReplyMsg((struct Message *)msg);
408    
409     // Action depends on message class
410     switch (iclass) {
411    
412     case IDCMP_CLOSEWINDOW: // Closing the window quits Frodo
413     TheC64->Quit();
414     break;
415    
416     case IDCMP_RAWKEY:
417     switch (code) {
418    
419     case 0x58: // F9: NMI (Restore)
420     TheC64->NMI();
421     break;
422    
423     case 0x59: // F10: Reset
424     TheC64->Reset();
425     break;
426    
427     case 0x5e: // '+' on keypad: Increase SkipFrames
428     ThePrefs.SkipFrames++;
429     break;
430    
431     case 0x4a: // '-' on keypad: Decrease SkipFrames
432     if (ThePrefs.SkipFrames > 1)
433     ThePrefs.SkipFrames--;
434     break;
435    
436     case 0x5d: // '*' on keypad: Toggle speed limiter
437     ThePrefs.LimitSpeed = !ThePrefs.LimitSpeed;
438     break;
439    
440     case 0x5c:{ // '/' on keypad: Toggle processor-level 1541 emulation
441     Prefs *prefs = new Prefs(ThePrefs);
442     prefs->Emul1541Proc = !prefs->Emul1541Proc;
443     TheC64->NewPrefs(prefs);
444     ThePrefs = *prefs;
445     delete prefs;
446     break;
447     }
448    
449     default:{
450     // Convert Amiga keycode to C64 row/column
451     int c64_byte = key_byte[code & 0x7f];
452     int c64_bit = key_bit[code & 0x7f];
453    
454     if (c64_byte != -1) {
455     if (!(c64_byte & 0x20)) {
456    
457     // Normal keys
458     bool shifted = c64_byte & 8;
459     c64_byte &= 7;
460     if (!(code & 0x80)) {
461    
462     // Key pressed
463     if (shifted) {
464     key_matrix[6] &= 0xef;
465     rev_matrix[4] &= 0xbf;
466     }
467     key_matrix[c64_byte] &= ~(1 << c64_bit);
468     rev_matrix[c64_bit] &= ~(1 << c64_byte);
469     } else {
470    
471     // Key released
472     if (shifted) {
473     key_matrix[6] |= 0x10;
474     rev_matrix[4] |= 0x40;
475     }
476     key_matrix[c64_byte] |= (1 << c64_bit);
477     rev_matrix[c64_bit] |= (1 << c64_byte);
478     }
479     } else {
480    
481     // Joystick emulation
482     c64_byte &= 0x1f;
483     if (code & 0x80)
484     *joystick |= c64_byte;
485     else
486     *joystick &= ~c64_byte;
487     }
488     }
489     }
490     }
491     break;
492    
493     case IDCMP_MENUPICK:{
494     if (code == MENUNULL)
495     break;
496    
497     // Get item number
498     int item_number = ITEMNUM(code);
499     switch (item_number) {
500    
501     case 0: { // About Frodo
502     TheC64->Pause();
503     char str[256];
504     sprintf(str, "%s by Christian Bauer\n<cbauer@iphcip1.physik.uni-mainz.de>\nİ Copyright 1994-1997\nFreely distributable", VERSION_STRING);
505     ShowRequester(str, "OK");
506     TheC64->Resume();
507     break;
508     }
509    
510     case 2: // Preferences
511     TheC64->Pause();
512     be_app->RunPrefsEditor();
513     TheC64->Resume();
514     break;
515    
516     case 4: // Reset C64
517     TheC64->Reset();
518     break;
519    
520     case 5: // Insert next disk
521     if (strlen(ThePrefs.DrivePath[0]) > 4) {
522     char str[256];
523     strcpy(str, ThePrefs.DrivePath[0]);
524     char *p = str + strlen(str) - 5;
525    
526     // If path matches "*.?64", increment character before the '.'
527     if (p[1] == '.' && p[3] == '6' && p[4] == '4') {
528     p[0]++;
529    
530     // If no such file exists, set character before the '.' to '1', 'a' or 'A'
531     FILE *file;
532     if ((file = fopen(str, "rb")) == NULL) {
533     if (isdigit(p[0]))
534     p[0] = '1';
535     else if (isupper(p[0]))
536     p[0] = 'A';
537     else
538     p[0] = 'a';
539     } else
540     fclose(file);
541    
542     // Set new prefs
543     Prefs *prefs = new Prefs(ThePrefs);
544     strcpy(prefs->DrivePath[0], str);
545     TheC64->NewPrefs(prefs);
546     ThePrefs = *prefs;
547     delete prefs;
548     }
549     }
550     break;
551    
552     case 6: // SAM
553     TheC64->Pause();
554     SAM(TheC64);
555     TheC64->Resume();
556     break;
557    
558     case 8: // Load snapshot
559     if (open_req != NULL && AslRequest(open_req, NULL)) {
560     char path[256];
561     strncpy(path, open_req->fr_Drawer, 255);
562     AddPart(path, open_req->fr_File, 255);
563     TheC64->Pause();
564     TheC64->LoadSnapshot(path);
565     TheC64->Resume();
566     }
567     break;
568    
569     case 9: // Save snapshot
570     if (save_req != NULL && AslRequest(save_req, NULL)) {
571     char path[256];
572     strncpy(path, save_req->fr_Drawer, 255);
573     AddPart(path, save_req->fr_File, 255);
574     TheC64->Pause();
575     TheC64->SaveSnapshot(path);
576     TheC64->Resume();
577     }
578     break;
579    
580     case 11: // Quit Frodo
581     TheC64->Quit();
582     break;
583     }
584     break;
585     }
586    
587     case IDCMP_REFRESHWINDOW:
588     BeginRefresh(the_window);
589     draw_led_bar();
590     EndRefresh(the_window, TRUE);
591     break;
592     }
593     }
594     }
595    
596    
597     /*
598     * Check if NumLock is down (for switching the joystick keyboard emulation)
599     */
600    
601     bool C64Display::NumLock(void)
602     {
603     return FALSE;
604     }
605    
606    
607     /*
608     * Allocate C64 colors
609     */
610    
611     void C64Display::InitColors(UBYTE *colors)
612     {
613     // Spread pens into colors array
614     for (int i=0; i<256; i++)
615     colors[i] = pens[i & 0x0f];
616     }
617    
618    
619     /*
620     * Show a requester
621     */
622    
623     long ShowRequester(char *str, char *button1, char *button2)
624     {
625     struct EasyStruct es;
626     char gads[256];
627    
628     strcpy(gads, button1);
629     if (button2) {
630     strcat(gads, "|");
631     strcat(gads, button2);
632     }
633    
634     es.es_StructSize = sizeof(struct EasyStruct);
635     es.es_Flags = 0;
636     es.es_Title = "Frodo";
637     es.es_TextFormat = str;
638     es.es_GadgetFormat = gads;
639    
640     return EasyRequestArgs(NULL, &es, NULL, NULL) % 1;
641     }