mirror of
				https://github.com/RaySollium99/libpicofe.git
				synced 2025-10-26 09:09:40 -04:00 
			
		
		
		
	work on 'vblank on line start' problem, var changes, mask defines
git-svn-id: file:///home/notaz/opt/svn/PicoDrive/platform@408 be3aeb3a-fb24-0410-a615-afba39da0efa
This commit is contained in:
		
							parent
							
								
									0fb424dbee
								
							
						
					
					
						commit
						dd5fd4778d
					
				
					 10 changed files with 58 additions and 725 deletions
				
			
		|  | @ -70,9 +70,9 @@ static void custom_write(FILE *f, const menu_entry *me, int no_def) | |||
| 	switch (me->id) | ||||
| 	{ | ||||
| 		case MA_OPT_RENDERER: | ||||
| 			if (no_def && !((defaultConfig.s_PicoOpt^PicoOpt)&0x10) && | ||||
| 			if (no_def && !((defaultConfig.s_PicoOpt^PicoOpt)&POPT_ALT_RENDERER) && | ||||
| 				!((defaultConfig.EmuOpt^currentConfig.EmuOpt)&0x80)) return; | ||||
| 			if (PicoOpt&0x10) | ||||
| 			if (PicoOpt&POPT_ALT_RENDERER) | ||||
| 				str = "8bit fast"; | ||||
| 			else if (currentConfig.EmuOpt&0x80) | ||||
| 				str = "16bit accurate"; | ||||
|  | @ -99,9 +99,9 @@ static void custom_write(FILE *f, const menu_entry *me, int no_def) | |||
| 			fprintf(f, "Frameskip = %s", str24); | ||||
| 			break; | ||||
| 		case MA_OPT_SOUND_QUALITY: | ||||
| 			if (no_def && !((defaultConfig.s_PicoOpt^PicoOpt)&8) && | ||||
| 			if (no_def && !((defaultConfig.s_PicoOpt^PicoOpt)&POPT_EN_STEREO) && | ||||
| 				defaultConfig.s_PsndRate == PsndRate) return; | ||||
| 			str = (PicoOpt&0x08)?"stereo":"mono"; | ||||
| 			str = (PicoOpt&POPT_EN_STEREO)?"stereo":"mono"; | ||||
| 			fprintf(f, "Sound Quality = %i %s", PsndRate, str); | ||||
| 			break; | ||||
| 		case MA_OPT_REGION: | ||||
|  | @ -438,14 +438,14 @@ static int custom_read(menu_entry *me, const char *var, const char *val) | |||
| 		case MA_OPT_RENDERER: | ||||
| 			if (strcasecmp(var, "Renderer") != 0) return 0; | ||||
| 			if      (strcasecmp(val, "8bit fast") == 0) { | ||||
| 				PicoOpt |=  0x10; | ||||
| 				PicoOpt |=  POPT_ALT_RENDERER; | ||||
| 			} | ||||
| 			else if (strcasecmp(val, "16bit accurate") == 0) { | ||||
| 				PicoOpt &= ~0x10; | ||||
| 				PicoOpt &= ~POPT_ALT_RENDERER; | ||||
| 				currentConfig.EmuOpt |=  0x80; | ||||
| 			} | ||||
| 			else if (strcasecmp(val, "8bit accurate") == 0) { | ||||
| 				PicoOpt &= ~0x10; | ||||
| 				PicoOpt &= ~POPT_ALT_RENDERER; | ||||
| 				currentConfig.EmuOpt &= ~0x80; | ||||
| 			} | ||||
| 			else | ||||
|  | @ -480,9 +480,9 @@ static int custom_read(menu_entry *me, const char *var, const char *val) | |||
| 				PsndRate = 22050; | ||||
| 			while (*tmp == ' ') tmp++; | ||||
| 			if        (strcasecmp(tmp, "stereo") == 0) { | ||||
| 				PicoOpt |=  8; | ||||
| 				PicoOpt |=  POPT_EN_STEREO; | ||||
| 			} else if (strcasecmp(tmp, "mono") == 0) { | ||||
| 				PicoOpt &= ~8; | ||||
| 				PicoOpt &= ~POPT_EN_STEREO; | ||||
| 			} else | ||||
| 				return 0; | ||||
| 			return 1; | ||||
|  |  | |||
							
								
								
									
										36
									
								
								common/emu.c
									
										
									
									
									
								
							
							
						
						
									
										36
									
								
								common/emu.c
									
										
									
									
									
								
							|  | @ -235,7 +235,7 @@ char *emu_makeRomId(void) | |||
| 	static char id_string[3+0x11+0x11+0x30+16]; | ||||
| 	int pos, swab = 1; | ||||
| 
 | ||||
| 	if (PicoMCD & 1) { | ||||
| 	if (PicoAHW & PAHW_MCD) { | ||||
| 		strcpy(id_string, "CD|"); | ||||
| 		swab = 0; | ||||
| 	} | ||||
|  | @ -325,14 +325,14 @@ int emu_ReloadRom(void) | |||
| 		get_ext(romFileName, ext); | ||||
| 	} | ||||
| 
 | ||||
| 	if ((PicoMCD & 1) && Pico_mcd != NULL) | ||||
| 	if ((PicoAHW & PAHW_MCD) && Pico_mcd != NULL) | ||||
| 		Stop_CD(); | ||||
| 
 | ||||
| 	// check for MegaCD image
 | ||||
| 	cd_state = emu_cdCheck(&cd_region); | ||||
| 	if (cd_state > 0) | ||||
| 	{ | ||||
| 		PicoMCD |= 1; | ||||
| 		PicoAHW |= PAHW_MCD; | ||||
| 		// valid CD image, check for BIOS..
 | ||||
| 
 | ||||
| 		// we need to have config loaded at this point
 | ||||
|  | @ -346,7 +346,7 @@ int emu_ReloadRom(void) | |||
| 		} | ||||
| 		if (!emu_findBios(cd_region, &used_rom_name)) { | ||||
| 			// bios_help() ?
 | ||||
| 			PicoMCD &= ~1; | ||||
| 			PicoAHW &= ~PAHW_MCD; | ||||
| 			return 0; | ||||
| 		} | ||||
| 
 | ||||
|  | @ -354,8 +354,8 @@ int emu_ReloadRom(void) | |||
| 	} | ||||
| 	else | ||||
| 	{ | ||||
| 		if (PicoMCD & 1) Stop_CD(); | ||||
| 		PicoMCD &= ~1; | ||||
| 		if (PicoAHW & PAHW_MCD) Stop_CD(); | ||||
| 		PicoAHW &= ~PAHW_MCD; | ||||
| 	} | ||||
| 
 | ||||
| 	rom = pm_open(used_rom_name); | ||||
|  | @ -388,7 +388,7 @@ int emu_ReloadRom(void) | |||
| 	} | ||||
| 
 | ||||
| 	// load config for this ROM (do this before insert to get correct region)
 | ||||
| 	if (!(PicoMCD&1)) | ||||
| 	if (!(PicoAHW & PAHW_MCD)) | ||||
| 		memcpy(id_header, rom_data + 0x100, sizeof(id_header)); | ||||
| 	if (!cfg_loaded) { | ||||
| 		ret = emu_ReadConfig(1, 1); | ||||
|  | @ -425,9 +425,9 @@ int emu_ReloadRom(void) | |||
| 	// additional movie stuff
 | ||||
| 	if (movie_data) { | ||||
| 		if(movie_data[0x14] == '6') | ||||
| 		     PicoOpt |=  0x20; // 6 button pad
 | ||||
| 		else PicoOpt &= ~0x20; | ||||
| 		PicoOpt |= 0x10040; // accurate timing, no VDP fifo timing
 | ||||
| 		     PicoOpt |=  POPT_6BTN_PAD; // 6 button pad
 | ||||
| 		else PicoOpt &= ~POPT_6BTN_PAD; | ||||
| 		PicoOpt |= POPT_DIS_VDP_FIFO|POPT_ACC_TIMING; // accurate timing, no VDP fifo timing
 | ||||
| 		if(movie_data[0xF] >= 'A') { | ||||
| 			if(movie_data[0x16] & 0x80) { | ||||
| 				PicoRegionOverride = 8; | ||||
|  | @ -442,7 +442,7 @@ int emu_ReloadRom(void) | |||
| 	} | ||||
| 	else | ||||
| 	{ | ||||
| 		PicoOpt &= ~0x10000; | ||||
| 		PicoOpt &= ~POPT_DIS_VDP_FIFO; | ||||
| 		if(Pico.m.pal) { | ||||
| 			strcpy(noticeMsg, "PAL SYSTEM / 50 FPS"); | ||||
| 		} else { | ||||
|  | @ -773,11 +773,11 @@ char *emu_GetSaveFName(int load, int is_sram, int slot) | |||
| 
 | ||||
| 	if (is_sram) | ||||
| 	{ | ||||
| 		romfname_ext(saveFname, (PicoMCD&1) ? "brm/" : "srm/", (PicoMCD&1) ? ".brm" : ".srm"); | ||||
| 		romfname_ext(saveFname, (PicoAHW&1) ? "brm/" : "srm/", (PicoAHW&1) ? ".brm" : ".srm"); | ||||
| 		if (load) { | ||||
| 			if (try_ropen_file(saveFname)) return saveFname; | ||||
| 			// try in current dir..
 | ||||
| 			romfname_ext(saveFname, NULL, (PicoMCD&1) ? ".brm" : ".srm"); | ||||
| 			romfname_ext(saveFname, NULL, (PicoAHW & PAHW_MCD) ? ".brm" : ".srm"); | ||||
| 			if (try_ropen_file(saveFname)) return saveFname; | ||||
| 			return NULL; // give up
 | ||||
| 		} | ||||
|  | @ -855,8 +855,9 @@ int emu_SaveLoadGame(int load, int sram) | |||
| 		int sram_size; | ||||
| 		unsigned char *sram_data; | ||||
| 		int truncate = 1; | ||||
| 		if (PicoMCD&1) { | ||||
| 			if (PicoOpt&0x8000) { // MCD RAM cart?
 | ||||
| 		if (PicoAHW & PAHW_MCD) | ||||
| 		{ | ||||
| 			if (PicoOpt&POPT_EN_MCD_RAMCART) { | ||||
| 				sram_size = 0x12000; | ||||
| 				sram_data = SRam.data; | ||||
| 				if (sram_data) | ||||
|  | @ -873,12 +874,13 @@ int emu_SaveLoadGame(int load, int sram) | |||
| 		} | ||||
| 		if (!sram_data) return 0; // SRam forcefully disabled for this game
 | ||||
| 
 | ||||
| 		if (load) { | ||||
| 		if (load) | ||||
| 		{ | ||||
| 			sramFile = fopen(saveFname, "rb"); | ||||
| 			if(!sramFile) return -1; | ||||
| 			fread(sram_data, 1, sram_size, sramFile); | ||||
| 			fclose(sramFile); | ||||
| 			if ((PicoMCD&1) && (PicoOpt&0x8000)) | ||||
| 			if ((PicoAHW & PAHW_MCD) && (PicoOpt&POPT_EN_MCD_RAMCART)) | ||||
| 				memcpy32((int *)Pico_mcd->bram, (int *)sram_data, 0x2000/4); | ||||
| 		} else { | ||||
| 			// sram save needs some special processing
 | ||||
|  |  | |||
|  | @ -180,7 +180,7 @@ void mp3_start_play(FILE *f, int pos) | |||
| 	mp3_current_file = NULL; | ||||
| 	mp3_buffer_offs = 0; | ||||
| 
 | ||||
| 	if (!(PicoOpt&0x800) || f == NULL) // cdda disabled or no file?
 | ||||
| 	if (!(PicoOpt&POPT_EN_MCD_CDDA) || f == NULL) // cdda disabled or no file?
 | ||||
| 		return; | ||||
| 
 | ||||
| 	//lprintf("mp3_start_play %p %i\n", f, pos);
 | ||||
|  | @ -204,7 +204,7 @@ int mp3_get_offset(void) | |||
| 	unsigned int offs1024 = 0; | ||||
| 	int cdda_on; | ||||
| 
 | ||||
| 	cdda_on = (PicoMCD & 1) && (PicoOpt&0x800) && !(Pico_mcd->s68k_regs[0x36] & 1) && | ||||
| 	cdda_on = (PicoAHW & PAHW_MCD) && (PicoOpt&POPT_EN_MCD_CDDA) && !(Pico_mcd->s68k_regs[0x36] & 1) && | ||||
| 			(Pico_mcd->scd.Status_CDC & 1) && mp3_current_file != NULL; | ||||
| 
 | ||||
| 	if (cdda_on) { | ||||
|  |  | |||
|  | @ -611,7 +611,7 @@ int mp3_get_offset(void) | |||
| 	unsigned int offs1024 = 0; | ||||
| 	int cdda_on; | ||||
| 
 | ||||
| 	cdda_on = (PicoMCD & 1) && (PicoOpt&0x800) && !(Pico_mcd->s68k_regs[0x36] & 1) && | ||||
| 	cdda_on = (PicoAHW & PAHW_MCD) && (PicoOpt&0x800) && !(Pico_mcd->s68k_regs[0x36] & 1) && | ||||
| 			(Pico_mcd->scd.Status_CDC & 1) && loaded_mp3; | ||||
| 
 | ||||
| 	if (cdda_on) { | ||||
|  |  | |||
							
								
								
									
										28
									
								
								gp2x/emu.c
									
										
									
									
									
								
							
							
						
						
									
										28
									
								
								gp2x/emu.c
									
										
									
									
									
								
							|  | @ -225,18 +225,18 @@ static void cd_leds(void) | |||
| 	} | ||||
| } | ||||
| 
 | ||||
| static int EmuScan16(unsigned int num, void *sdata) | ||||
| static int EmuScanBegin16(unsigned int num) | ||||
| { | ||||
| 	if (!(Pico.video.reg[1]&8)) num += 8; | ||||
| 	DrawLineDest = (unsigned short *) gp2x_screen + 320*(num+1); | ||||
| 	DrawLineDest = (unsigned short *) gp2x_screen + 320 * num; | ||||
| 
 | ||||
| 	return 0; | ||||
| } | ||||
| 
 | ||||
| static int EmuScan8(unsigned int num, void *sdata) | ||||
| static int EmuScanBegin8(unsigned int num) | ||||
| { | ||||
| 	if (!(Pico.video.reg[1]&8)) num += 8; | ||||
| 	DrawLineDest = (unsigned char *)  gp2x_screen + 320*(num+1); | ||||
| 	DrawLineDest = (unsigned char *)  gp2x_screen + 320 * num; | ||||
| 
 | ||||
| 	return 0; | ||||
| } | ||||
|  | @ -266,7 +266,8 @@ static void blit(const char *fps, const char *notice) | |||
| 	else if (!(emu_opt&0x80)) | ||||
| 	{ | ||||
| 		// 8bit accurate renderer
 | ||||
| 		if (Pico.m.dirtyPal) { | ||||
| 		if (Pico.m.dirtyPal) | ||||
| 		{ | ||||
| 			Pico.m.dirtyPal = 0; | ||||
| 			if(Pico.video.reg[0xC]&8) { // shadow/hilight mode
 | ||||
| 				vidConvCpyRGB32(localPal, Pico.cram, 0x40); | ||||
|  | @ -278,7 +279,7 @@ static void blit(const char *fps, const char *notice) | |||
| 				localPal[0xe0] = 0x00000000; // reserved pixels for OSD
 | ||||
| 				localPal[0xf0] = 0x00ffffff; | ||||
| 				gp2x_video_setpalette(localPal, 0x100); | ||||
| 			} else if (rendstatus & 0x20) { // mid-frame palette changes
 | ||||
| 			} else if (rendstatus & PDRAW_SONIC_MODE) { // mid-frame palette changes
 | ||||
| 				vidConvCpyRGB32(localPal, Pico.cram, 0x40); | ||||
| 				vidConvCpyRGB32(localPal+0x40, HighPal, 0x40); | ||||
| 				vidConvCpyRGB32(localPal+0x80, HighPal+0x40, 0x40); | ||||
|  | @ -297,7 +298,7 @@ static void blit(const char *fps, const char *notice) | |||
| 		if (emu_opt & 2) | ||||
| 			osd_text(osd_fps_x, h, fps); | ||||
| 	} | ||||
| 	if ((emu_opt & 0x400) && (PicoMCD & 1)) | ||||
| 	if ((emu_opt & 0x400) && (PicoAHW & PAHW_MCD)) | ||||
| 		cd_leds(); | ||||
| 
 | ||||
| 	//gp2x_video_wait_vsync();
 | ||||
|  | @ -339,13 +340,11 @@ static void vidResetMode(void) | |||
| 	} else if (currentConfig.EmuOpt&0x80) { | ||||
| 		gp2x_video_changemode(16); | ||||
| 		PicoDrawSetColorFormat(1); | ||||
| 		PicoScan = EmuScan16; | ||||
| 		PicoScan(0, 0); | ||||
| 		PicoScanBegin = EmuScanBegin16; | ||||
| 	} else { | ||||
| 		gp2x_video_changemode(8); | ||||
| 		PicoDrawSetColorFormat(2); | ||||
| 		PicoScan = EmuScan8; | ||||
| 		PicoScan(0, 0); | ||||
| 		PicoScanBegin = EmuScanBegin8; | ||||
| 	} | ||||
| 	if ((PicoOpt&0x10)||!(currentConfig.EmuOpt&0x80)) { | ||||
| 		// setup pal for 8-bit modes
 | ||||
|  | @ -625,8 +624,7 @@ void emu_forcedFrame(void) | |||
| 
 | ||||
| 	//vidResetMode();
 | ||||
| 	PicoDrawSetColorFormat(1); | ||||
| 	PicoScan = EmuScan16; | ||||
| 	PicoScan(0, 0); | ||||
| 	PicoScanBegin = EmuScanBegin16; | ||||
| 	Pico.m.dirtyPal = 1; | ||||
| 	PicoFrameDrawOnly(); | ||||
| 
 | ||||
|  | @ -782,7 +780,7 @@ void emu_Loop(void) | |||
| 	} | ||||
| 
 | ||||
| 	// prepare CD buffer
 | ||||
| 	if (PicoMCD & 1) PicoCDBufferInit(); | ||||
| 	if (PicoAHW & PAHW_MCD) PicoCDBufferInit(); | ||||
| 
 | ||||
| 	// calc vsync offset to sync timing code with vsync
 | ||||
| 	if (currentConfig.EmuOpt&0x2000) { | ||||
|  | @ -960,7 +958,7 @@ void emu_Loop(void) | |||
| 
 | ||||
| 	change_fast_forward(0); | ||||
| 
 | ||||
| 	if (PicoMCD & 1) PicoCDBufferFree(); | ||||
| 	if (PicoAHW & PAHW_MCD) PicoCDBufferFree(); | ||||
| 
 | ||||
| 	// save SRAM
 | ||||
| 	if((currentConfig.EmuOpt & 1) && SRam.changed) { | ||||
|  |  | |||
							
								
								
									
										10
									
								
								gp2x/menu.c
									
										
									
									
									
								
							
							
						
						
									
										10
									
								
								gp2x/menu.c
									
										
									
									
									
								
							|  | @ -553,7 +553,7 @@ static void draw_savestate_bg(int slot) | |||
| 	} | ||||
| 
 | ||||
| 	if (file) { | ||||
| 		if (PicoMCD & 1) { | ||||
| 		if (PicoAHW & PAHW_MCD) { | ||||
| 			PicoCdLoadStateGfx(file); | ||||
| 		} else { | ||||
| 			areaSeek(file, 0x10020, SEEK_SET);  // skip header and RAM in state file
 | ||||
|  | @ -878,7 +878,7 @@ static void kc_sel_loop(void) | |||
| { | ||||
| 	int menu_sel = 3, menu_sel_max = 3; | ||||
| 	unsigned long inp = 0; | ||||
| 	int is_6button = PicoOpt & 0x020; | ||||
| 	int is_6button = PicoOpt & POPT_6BTN_PAD; | ||||
| 
 | ||||
| 	while (1) | ||||
| 	{ | ||||
|  | @ -1174,7 +1174,7 @@ static void menu_opt_cust_draw(const menu_entry *entry, int x, int y, void *para | |||
| 	switch (entry->id) | ||||
| 	{ | ||||
| 		case MA_OPT_RENDERER: | ||||
| 			if (PicoOpt&0x10) | ||||
| 			if (PicoOpt & POPT_ALT_RENDERER) | ||||
| 				str = " 8bit fast"; | ||||
| 			else if (currentConfig.EmuOpt&0x80) | ||||
| 				str = "16bit accurate"; | ||||
|  | @ -1198,7 +1198,7 @@ static void menu_opt_cust_draw(const menu_entry *entry, int x, int y, void *para | |||
| 			text_out16(x, y, "Frameskip                  %s", str24); | ||||
| 			break; | ||||
| 		case MA_OPT_SOUND_QUALITY: | ||||
| 			str = (PicoOpt&0x08)?"stereo":"mono"; | ||||
| 			str = (PicoOpt & POPT_EN_STEREO) ? "stereo" : "mono"; | ||||
| 			text_out16(x, y, "Sound Quality:     %5iHz %s", PsndRate, str); | ||||
| 			break; | ||||
| 		case MA_OPT_REGION: | ||||
|  | @ -1288,7 +1288,7 @@ static void menu_options_save(void) | |||
| 		// force setting possibly changed..
 | ||||
| 		Pico.m.pal = (PicoRegionOverride == 2 || PicoRegionOverride == 8) ? 1 : 0; | ||||
| 	} | ||||
| 	if (!(PicoOpt & 0x20)) { | ||||
| 	if (!(PicoOpt & POPT_6BTN_PAD)) { | ||||
| 		// unbind XYZ MODE, just in case
 | ||||
| 		unbind_action(0xf00, -1, -1); | ||||
| 	} | ||||
|  |  | |||
							
								
								
									
										667
									
								
								linux/fakedasm.c
									
										
									
									
									
								
							
							
						
						
									
										667
									
								
								linux/fakedasm.c
									
										
									
									
									
								
							|  | @ -1,667 +0,0 @@ | |||
| // This is part of Pico Library
 | ||||
| 
 | ||||
| // (c) Copyright 2004 Dave, All rights reserved.
 | ||||
| // (c) Copyright 2006 notaz, All rights reserved.
 | ||||
| // Free for non-commercial use.
 | ||||
| 
 | ||||
| // For commercial use, separate licencing terms must be obtained.
 | ||||
| 
 | ||||
| 
 | ||||
| #include "../../Pico/PicoInt.h" | ||||
| #undef blockcpy | ||||
| 
 | ||||
| extern unsigned short DefOutBuff[320*2]; | ||||
| extern unsigned char  HighCol[8+320+8]; | ||||
| extern char HighSprZ[320+8+8]; // Z-buffer for accurate sprites and shadow/hilight mode
 | ||||
|                         // (if bit 7 == 0, sh caused by tile; if bit 6 == 0 pixel must be shadowed, else hilighted, if bit5 == 1)
 | ||||
| // lsb->msb: moved sprites, all window tiles don't use same priority, accurate sprites (copied from PicoOpt), interlace
 | ||||
| //           dirty sprites, sonic mode
 | ||||
| extern int rendstatus; | ||||
| extern int Scanline; // Scanline
 | ||||
| 
 | ||||
| 
 | ||||
| struct TileStrip | ||||
| { | ||||
|   int nametab; // Position in VRAM of name table (for this tile line)
 | ||||
|   int line;    // Line number in pixels 0x000-0x3ff within the virtual tilemap
 | ||||
|   int hscroll; // Horizontal scroll value in pixels for the line
 | ||||
|   int xmask;   // X-Mask (0x1f - 0x7f) for horizontal wraparound in the tilemap
 | ||||
|   int *hc;     // cache for high tile codes and their positions
 | ||||
|   int cells;   // cells (tiles) to draw (32 col mode doesn't need to update whole 320)
 | ||||
| }; | ||||
| 
 | ||||
| // utility
 | ||||
| void *blockcpy(void *dst, const void *src, size_t n) | ||||
| { | ||||
| 	return memcpy(dst, src, n); | ||||
| } | ||||
| 
 | ||||
| void blockcpy_or(void *dst, void *src, size_t n, int pat) | ||||
| { | ||||
|   unsigned char *pd = dst, *ps = src; | ||||
|   for (; n; n--) | ||||
|     *pd++ = (unsigned char) (*ps++ | pat); | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| static int TileNorm(int sx,int addr,int pal) | ||||
| { | ||||
|   unsigned char *pd = HighCol+sx; | ||||
|   unsigned int pack=0; unsigned int t=0; | ||||
| 
 | ||||
|   pack=*(unsigned int *)(Pico.vram+addr); // Get 8 pixels
 | ||||
|   if (pack) | ||||
|   { | ||||
|     t=pack&0x0000f000; if (t) pd[0]=(unsigned char)(pal|(t>>12)); | ||||
|     t=pack&0x00000f00; if (t) pd[1]=(unsigned char)(pal|(t>> 8)); | ||||
|     t=pack&0x000000f0; if (t) pd[2]=(unsigned char)(pal|(t>> 4)); | ||||
|     t=pack&0x0000000f; if (t) pd[3]=(unsigned char)(pal|(t    )); | ||||
|     t=pack&0xf0000000; if (t) pd[4]=(unsigned char)(pal|(t>>28)); | ||||
|     t=pack&0x0f000000; if (t) pd[5]=(unsigned char)(pal|(t>>24)); | ||||
|     t=pack&0x00f00000; if (t) pd[6]=(unsigned char)(pal|(t>>20)); | ||||
|     t=pack&0x000f0000; if (t) pd[7]=(unsigned char)(pal|(t>>16)); | ||||
|     return 0; | ||||
|   } | ||||
| 
 | ||||
|   return 1; // Tile blank
 | ||||
| } | ||||
| 
 | ||||
| static int TileFlip(int sx,int addr,int pal) | ||||
| { | ||||
|   unsigned char *pd = HighCol+sx; | ||||
|   unsigned int pack=0; unsigned int t=0; | ||||
| 
 | ||||
|   pack=*(unsigned int *)(Pico.vram+addr); // Get 8 pixels
 | ||||
|   if (pack) | ||||
|   { | ||||
|     t=pack&0x000f0000; if (t) pd[0]=(unsigned char)(pal|(t>>16)); | ||||
|     t=pack&0x00f00000; if (t) pd[1]=(unsigned char)(pal|(t>>20)); | ||||
|     t=pack&0x0f000000; if (t) pd[2]=(unsigned char)(pal|(t>>24)); | ||||
|     t=pack&0xf0000000; if (t) pd[3]=(unsigned char)(pal|(t>>28)); | ||||
|     t=pack&0x0000000f; if (t) pd[4]=(unsigned char)(pal|(t    )); | ||||
|     t=pack&0x000000f0; if (t) pd[5]=(unsigned char)(pal|(t>> 4)); | ||||
|     t=pack&0x00000f00; if (t) pd[6]=(unsigned char)(pal|(t>> 8)); | ||||
|     t=pack&0x0000f000; if (t) pd[7]=(unsigned char)(pal|(t>>12)); | ||||
|     return 0; | ||||
|   } | ||||
|   return 1; // Tile blank
 | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| // tile renderers for hacky operator sprite support
 | ||||
| #define sh_pix(x) \ | ||||
|   if(!t); \ | ||||
|   else if(t==0xe) pd[x]=(unsigned char)((pd[x]&0x3f)|0x80); /* hilight */ \ | ||||
|   else if(t==0xf) pd[x]=(unsigned char)((pd[x]&0x3f)|0xc0); /* shadow  */ \ | ||||
|   else pd[x]=(unsigned char)(pal|t); | ||||
| 
 | ||||
| static int TileNormSH(int sx,int addr,int pal) | ||||
| { | ||||
|   unsigned int pack=0; unsigned int t=0; | ||||
|   unsigned char *pd = HighCol+sx; | ||||
| 
 | ||||
|   pack=*(unsigned int *)(Pico.vram+addr); // Get 8 pixels
 | ||||
|   if (pack) | ||||
|   { | ||||
|     t=(pack&0x0000f000)>>12; sh_pix(0); | ||||
|     t=(pack&0x00000f00)>> 8; sh_pix(1); | ||||
|     t=(pack&0x000000f0)>> 4; sh_pix(2); | ||||
|     t=(pack&0x0000000f)    ; sh_pix(3); | ||||
|     t=(pack&0xf0000000)>>28; sh_pix(4); | ||||
|     t=(pack&0x0f000000)>>24; sh_pix(5); | ||||
|     t=(pack&0x00f00000)>>20; sh_pix(6); | ||||
|     t=(pack&0x000f0000)>>16; sh_pix(7); | ||||
|     return 0; | ||||
|   } | ||||
| 
 | ||||
|   return 1; // Tile blank
 | ||||
| } | ||||
| 
 | ||||
| static int TileFlipSH(int sx,int addr,int pal) | ||||
| { | ||||
|   unsigned int pack=0; unsigned int t=0; | ||||
|   unsigned char *pd = HighCol+sx; | ||||
| 
 | ||||
|   pack=*(unsigned int *)(Pico.vram+addr); // Get 8 pixels
 | ||||
|   if (pack) | ||||
|   { | ||||
|     t=(pack&0x000f0000)>>16; sh_pix(0); | ||||
|     t=(pack&0x00f00000)>>20; sh_pix(1); | ||||
|     t=(pack&0x0f000000)>>24; sh_pix(2); | ||||
|     t=(pack&0xf0000000)>>28; sh_pix(3); | ||||
|     t=(pack&0x0000000f)    ; sh_pix(4); | ||||
|     t=(pack&0x000000f0)>> 4; sh_pix(5); | ||||
|     t=(pack&0x00000f00)>> 8; sh_pix(6); | ||||
|     t=(pack&0x0000f000)>>12; sh_pix(7); | ||||
|     return 0; | ||||
|   } | ||||
|   return 1; // Tile blank
 | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| // --------------------------------------------
 | ||||
| 
 | ||||
| static void DrawStrip(struct TileStrip *ts, int sh) | ||||
| { | ||||
|   int tilex=0,dx=0,ty=0,code=0,addr=0,cells; | ||||
|   int oldcode=-1,blank=-1; // The tile we know is blank
 | ||||
|   int pal=0; | ||||
| 
 | ||||
|   // Draw tiles across screen:
 | ||||
|   tilex=(-ts->hscroll)>>3; | ||||
|   ty=(ts->line&7)<<1; // Y-Offset into tile
 | ||||
|   dx=((ts->hscroll-1)&7)+1; | ||||
|   cells = ts->cells; | ||||
|   if(dx != 8) cells++; // have hscroll, need to draw 1 cell more
 | ||||
| 
 | ||||
|   for (; cells; dx+=8,tilex++,cells--) | ||||
|   { | ||||
|     int zero=0; | ||||
| 
 | ||||
|     code=Pico.vram[ts->nametab+(tilex&ts->xmask)]; | ||||
|     if (code==blank) continue; | ||||
|     if (code>>15) { // high priority tile
 | ||||
|       int cval = code | (dx<<16) | (ty<<25); | ||||
|       if(code&0x1000) cval^=7<<26; | ||||
|       *ts->hc++ = cval; // cache it
 | ||||
|       continue; | ||||
|     } | ||||
| continue; | ||||
|     if (code!=oldcode) { | ||||
|       oldcode = code; | ||||
|       // Get tile address/2:
 | ||||
|       addr=(code&0x7ff)<<4; | ||||
|       addr+=ty; | ||||
|       if (code&0x1000) addr^=0xe; // Y-flip
 | ||||
| 
 | ||||
| //      pal=Pico.cram+((code>>9)&0x30);
 | ||||
|       pal=((code>>9)&0x30)|(sh<<6); | ||||
|     } | ||||
| 
 | ||||
|     if (code&0x0800) zero=TileFlip(dx,addr,pal); | ||||
|     else             zero=TileNorm(dx,addr,pal); | ||||
| 
 | ||||
|     if (zero) blank=code; // We know this tile is blank now
 | ||||
|   } | ||||
| 
 | ||||
|   // terminate the cache list
 | ||||
|   *ts->hc = 0; | ||||
| } | ||||
| 
 | ||||
| static void DrawStripVSRam(struct TileStrip *ts, int plane) | ||||
| { | ||||
|   int tilex=0,dx=0,ty=0,code=0,addr=0,cell=0,nametabadd=0; | ||||
|   int oldcode=-1,blank=-1; // The tile we know is blank
 | ||||
|   int pal=0,scan=Scanline; | ||||
| 
 | ||||
|   // Draw tiles across screen:
 | ||||
|   tilex=(-ts->hscroll)>>3; | ||||
|   dx=((ts->hscroll-1)&7)+1; | ||||
|   if(dx != 8) { | ||||
|     int vscroll, line; | ||||
|     cell--; // have hscroll, start with negative cell
 | ||||
|     // also calculate intial VS stuff
 | ||||
|     vscroll=Pico.vsram[plane]; | ||||
| 
 | ||||
|     // Find the line in the name table
 | ||||
|     line=(vscroll+scan)&ts->line&0xffff; // ts->line is really ymask ..
 | ||||
|     nametabadd=(line>>3)<<(ts->line>>24);    // .. and shift[width]
 | ||||
|     ty=(line&7)<<1; // Y-Offset into tile
 | ||||
|   } | ||||
| 
 | ||||
|   for (; cell < ts->cells; dx+=8,tilex++,cell++) | ||||
|   { | ||||
|     int zero=0; | ||||
| 
 | ||||
|     if((cell&1)==0) { | ||||
|       int line,vscroll; | ||||
|       vscroll=Pico.vsram[plane+(cell&~1)]; | ||||
| 
 | ||||
|       // Find the line in the name table
 | ||||
|       line=(vscroll+scan)&ts->line&0xffff; // ts->line is really ymask ..
 | ||||
|       nametabadd=(line>>3)<<(ts->line>>24);    // .. and shift[width]
 | ||||
|       ty=(line&7)<<1; // Y-Offset into tile
 | ||||
|     } | ||||
| 
 | ||||
|     code=Pico.vram[ts->nametab+nametabadd+(tilex&ts->xmask)]; | ||||
|     if (code==blank) continue; | ||||
|     if (code>>15) { // high priority tile
 | ||||
|       int cval = code | (dx<<16) | (ty<<25); | ||||
|       if(code&0x1000) cval^=7<<26; | ||||
|       *ts->hc++ = cval; // cache it
 | ||||
|       continue; | ||||
|     } | ||||
| 
 | ||||
|     if (code!=oldcode) { | ||||
|       oldcode = code; | ||||
|       // Get tile address/2:
 | ||||
|       addr=(code&0x7ff)<<4; | ||||
|       if (code&0x1000) addr+=14-ty; else addr+=ty; // Y-flip
 | ||||
| 
 | ||||
| //      pal=Pico.cram+((code>>9)&0x30);
 | ||||
|       pal=((code>>9)&0x30); | ||||
|     } | ||||
| 
 | ||||
|     if (code&0x0800) zero=TileFlip(dx,addr,pal); | ||||
|     else             zero=TileNorm(dx,addr,pal); | ||||
| 
 | ||||
|     if (zero) blank=code; // We know this tile is blank now
 | ||||
|   } | ||||
| 
 | ||||
|   // terminate the cache list
 | ||||
|   *ts->hc = 0; | ||||
| } | ||||
| 
 | ||||
| static void DrawStripInterlace(struct TileStrip *ts) | ||||
| { | ||||
|   int tilex=0,dx=0,ty=0,code=0,addr=0,cells; | ||||
|   int oldcode=-1,blank=-1; // The tile we know is blank
 | ||||
|   int pal=0; | ||||
| 
 | ||||
|   // Draw tiles across screen:
 | ||||
|   tilex=(-ts->hscroll)>>3; | ||||
|   ty=(ts->line&15)<<1; // Y-Offset into tile
 | ||||
|   dx=((ts->hscroll-1)&7)+1; | ||||
|   cells = ts->cells; | ||||
|   if(dx != 8) cells++; // have hscroll, need to draw 1 cell more
 | ||||
| 
 | ||||
|   for (; cells; dx+=8,tilex++,cells--) | ||||
|   { | ||||
|     int zero=0; | ||||
| 
 | ||||
|     code=Pico.vram[ts->nametab+(tilex&ts->xmask)]; | ||||
|     if (code==blank) continue; | ||||
|     if (code>>15) { // high priority tile
 | ||||
|       int cval = (code&0xfc00) | (dx<<16) | (ty<<25); | ||||
|       cval|=(code&0x3ff)<<1; | ||||
|       if(code&0x1000) cval^=0xf<<26; | ||||
|       *ts->hc++ = cval; // cache it
 | ||||
|       continue; | ||||
|     } | ||||
| 
 | ||||
|     if (code!=oldcode) { | ||||
|       oldcode = code; | ||||
|       // Get tile address/2:
 | ||||
|       addr=(code&0x7ff)<<5; | ||||
|       if (code&0x1000) addr+=30-ty; else addr+=ty; // Y-flip
 | ||||
| 
 | ||||
| //      pal=Pico.cram+((code>>9)&0x30);
 | ||||
|       pal=((code>>9)&0x30); | ||||
|     } | ||||
| 
 | ||||
|     if (code&0x0800) zero=TileFlip(dx,addr,pal); | ||||
|     else             zero=TileNorm(dx,addr,pal); | ||||
| 
 | ||||
|     if (zero) blank=code; // We know this tile is blank now
 | ||||
|   } | ||||
| 
 | ||||
|   // terminate the cache list
 | ||||
|   *ts->hc = 0; | ||||
| } | ||||
| 
 | ||||
| // --------------------------------------------
 | ||||
| 
 | ||||
| void DrawLayer(int plane, int *hcache, int maxcells, int sh) | ||||
| { | ||||
|   struct PicoVideo *pvid=&Pico.video; | ||||
|   const char shift[4]={5,6,5,7}; // 32,64 or 128 sized tilemaps (2 is invalid)
 | ||||
|   struct TileStrip ts; | ||||
|   int width, height, ymask; | ||||
|   int vscroll, htab; | ||||
| 
 | ||||
|   ts.hc=hcache; | ||||
|   ts.cells=maxcells; | ||||
| 
 | ||||
|   // Work out the TileStrip to draw
 | ||||
| 
 | ||||
|   // Work out the name table size: 32 64 or 128 tiles (0-3)
 | ||||
|   width=pvid->reg[16]; | ||||
|   height=(width>>4)&3; width&=3; | ||||
| 
 | ||||
|   ts.xmask=(1<<shift[width])-1; // X Mask in tiles (0x1f-0x7f)
 | ||||
|   ymask=(height<<8)|0xff;       // Y Mask in pixels
 | ||||
|   if(width == 1)   ymask&=0x1ff; | ||||
|   else if(width>1) ymask =0x0ff; | ||||
| 
 | ||||
|   // Find name table:
 | ||||
|   if (plane==0) ts.nametab=(pvid->reg[2]&0x38)<< 9; // A
 | ||||
|   else          ts.nametab=(pvid->reg[4]&0x07)<<12; // B
 | ||||
| 
 | ||||
|   htab=pvid->reg[13]<<9; // Horizontal scroll table address
 | ||||
|   if ( pvid->reg[11]&2)     htab+=Scanline<<1; // Offset by line
 | ||||
|   if ((pvid->reg[11]&1)==0) htab&=~0xf; // Offset by tile
 | ||||
|   htab+=plane; // A or B
 | ||||
| 
 | ||||
|   // Get horizontal scroll value, will be masked later
 | ||||
|   ts.hscroll=Pico.vram[htab&0x7fff]; | ||||
| 
 | ||||
|   if((pvid->reg[12]&6) == 6) { | ||||
|     // interlace mode 2
 | ||||
|     vscroll=Pico.vsram[plane]; // Get vertical scroll value
 | ||||
| 
 | ||||
|     // Find the line in the name table
 | ||||
|     ts.line=(vscroll+(Scanline<<1))&((ymask<<1)|1); | ||||
|     ts.nametab+=(ts.line>>4)<<shift[width]; | ||||
| 
 | ||||
|     DrawStripInterlace(&ts); | ||||
|   } else if( pvid->reg[11]&4) { | ||||
|     // shit, we have 2-cell column based vscroll
 | ||||
|     // luckily this doesn't happen too often
 | ||||
|     ts.line=ymask|(shift[width]<<24); // save some stuff instead of line
 | ||||
|     DrawStripVSRam(&ts, plane); | ||||
|   } else { | ||||
|     vscroll=Pico.vsram[plane]; // Get vertical scroll value
 | ||||
| 
 | ||||
|     // Find the line in the name table
 | ||||
|     ts.line=(vscroll+Scanline)&ymask; | ||||
|     ts.nametab+=(ts.line>>3)<<shift[width]; | ||||
| 
 | ||||
|     DrawStrip(&ts, sh); | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| // --------------------------------------------
 | ||||
| 
 | ||||
| // tstart & tend are tile pair numbers
 | ||||
| void DrawWindow(int tstart, int tend, int prio, int sh) // int *hcache
 | ||||
| { | ||||
|   struct PicoVideo *pvid=&Pico.video; | ||||
|   int tilex=0,ty=0,nametab,code=0; | ||||
|   int blank=-1; // The tile we know is blank
 | ||||
| 
 | ||||
|   // Find name table line:
 | ||||
|   if (pvid->reg[12]&1) | ||||
|   { | ||||
|     nametab=(pvid->reg[3]&0x3c)<<9; // 40-cell mode
 | ||||
|     nametab+=(Scanline>>3)<<6; | ||||
|   } | ||||
|   else | ||||
|   { | ||||
|     nametab=(pvid->reg[3]&0x3e)<<9; // 32-cell mode
 | ||||
|     nametab+=(Scanline>>3)<<5; | ||||
|   } | ||||
| 
 | ||||
|   tilex=tstart<<1; | ||||
|   tend<<=1; | ||||
| 
 | ||||
|   ty=(Scanline&7)<<1; // Y-Offset into tile
 | ||||
| 
 | ||||
|   if(!(rendstatus&2)) { | ||||
|     // check the first tile code
 | ||||
|     code=Pico.vram[nametab+tilex]; | ||||
|     // if the whole window uses same priority (what is often the case), we may be able to skip this field
 | ||||
|     if((code>>15) != prio) return; | ||||
|   } | ||||
| 
 | ||||
|   // Draw tiles across screen:
 | ||||
|   for (; tilex < tend; tilex++) | ||||
|   { | ||||
|     int addr=0,zero=0; | ||||
|     int pal; | ||||
| 
 | ||||
|     code=Pico.vram[nametab+tilex]; | ||||
|     if(code==blank) continue; | ||||
|     if((code>>15) != prio) { | ||||
|       rendstatus|=2; | ||||
|       continue; | ||||
|     } | ||||
| 
 | ||||
|     pal=((code>>9)&0x30); | ||||
| 
 | ||||
|     if(sh) { | ||||
|       int tmp, *zb = (int *)(HighCol+8+(tilex<<3)); | ||||
|       if(prio) { | ||||
|         tmp = *zb; | ||||
|         if(!(tmp&0x00000080)) tmp&=~0x000000c0; if(!(tmp&0x00008000)) tmp&=~0x0000c000; | ||||
|         if(!(tmp&0x00800000)) tmp&=~0x00c00000; if(!(tmp&0x80000000)) tmp&=~0xc0000000; | ||||
|         *zb++=tmp; tmp = *zb; | ||||
|         if(!(tmp&0x00000080)) tmp&=~0x000000c0; if(!(tmp&0x00008000)) tmp&=~0x0000c000; | ||||
|         if(!(tmp&0x00800000)) tmp&=~0x00c00000; if(!(tmp&0x80000000)) tmp&=~0xc0000000; | ||||
|         *zb++=tmp; | ||||
|       } else { | ||||
|         pal |= 0x40; | ||||
|       } | ||||
|     } | ||||
| 
 | ||||
|     // Get tile address/2:
 | ||||
|     addr=(code&0x7ff)<<4; | ||||
|     if (code&0x1000) addr+=14-ty; else addr+=ty; // Y-flip
 | ||||
| 
 | ||||
|     if (code&0x0800) zero=TileFlip(8+(tilex<<3),addr,pal); | ||||
|     else             zero=TileNorm(8+(tilex<<3),addr,pal); | ||||
| 
 | ||||
|     if (zero) blank=code; // We know this tile is blank now
 | ||||
|   } | ||||
| 
 | ||||
|   // terminate the cache list
 | ||||
|   //*hcache = 0;
 | ||||
| } | ||||
| 
 | ||||
| // --------------------------------------------
 | ||||
| 
 | ||||
| void DrawTilesFromCache(int *hc, int sh) | ||||
| { | ||||
|   int code, addr, zero, dx; | ||||
|   int pal; | ||||
|   short blank=-1; // The tile we know is blank
 | ||||
| 
 | ||||
|   // *ts->hc++ = code | (dx<<16) | (ty<<25); // cache it
 | ||||
| 
 | ||||
|   while((code=*hc++)) { | ||||
|     if(!sh && (short)code == blank) continue; | ||||
| 
 | ||||
|     // Get tile address/2:
 | ||||
|     addr=(code&0x7ff)<<4; | ||||
|     addr+=(unsigned int)code>>25; // y offset into tile
 | ||||
|     dx=(code>>16)&0x1ff; | ||||
|     if(sh) { | ||||
|       unsigned char *zb = HighCol+dx; | ||||
|       if(!(*zb&0x80)) *zb&=0x3f; zb++; if(!(*zb&0x80)) *zb&=0x3f; zb++; | ||||
|       if(!(*zb&0x80)) *zb&=0x3f; zb++; if(!(*zb&0x80)) *zb&=0x3f; zb++; | ||||
|       if(!(*zb&0x80)) *zb&=0x3f; zb++; if(!(*zb&0x80)) *zb&=0x3f; zb++; | ||||
|       if(!(*zb&0x80)) *zb&=0x3f; zb++; if(!(*zb&0x80)) *zb&=0x3f; zb++; | ||||
|     } | ||||
| 
 | ||||
|     pal=((code>>9)&0x30); | ||||
| 
 | ||||
|     if (code&0x0800) zero=TileFlip(dx,addr,pal); | ||||
|     else             zero=TileNorm(dx,addr,pal); | ||||
| 
 | ||||
|     if(zero) blank=(short)code; | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| // --------------------------------------------
 | ||||
| 
 | ||||
| // Index + 0  :    hhhhvvvv ab--hhvv yyyyyyyy yyyyyyyy // a: offscreen h, b: offs. v, h: horiz. size
 | ||||
| // Index + 4  :    xxxxxxxx xxxxxxxx pccvhnnn nnnnnnnn // x: x coord + 8
 | ||||
| 
 | ||||
| void DrawSprite(int *sprite, int **hc, int sh) | ||||
| { | ||||
|   int width=0,height=0; | ||||
|   int row=0,code=0; | ||||
|   int pal; | ||||
|   int tile=0,delta=0; | ||||
|   int sx, sy; | ||||
|   int (*fTileFunc)(int sx,int addr,int pal); | ||||
| 
 | ||||
|   // parse the sprite data
 | ||||
|   sy=sprite[0]; | ||||
|   code=sprite[1]; | ||||
|   sx=code>>16; // X
 | ||||
|   width=sy>>28; | ||||
|   height=(sy>>24)&7; // Width and height in tiles
 | ||||
|   sy=(sy<<16)>>16; // Y
 | ||||
| 
 | ||||
|   row=Scanline-sy; // Row of the sprite we are on
 | ||||
| 
 | ||||
|   if (code&0x1000) row=(height<<3)-1-row; // Flip Y
 | ||||
| 
 | ||||
|   tile=code&0x7ff; // Tile number
 | ||||
|   tile+=row>>3; // Tile number increases going down
 | ||||
|   delta=height; // Delta to increase tile by going right
 | ||||
|   if (code&0x0800) { tile+=delta*(width-1); delta=-delta; } // Flip X
 | ||||
| 
 | ||||
|   tile<<=4; tile+=(row&7)<<1; // Tile address
 | ||||
| 
 | ||||
|   if(code&0x8000) { // high priority - cache it
 | ||||
|     *(*hc)++ = (tile<<16)|((code&0x0800)<<5)|((sx<<6)&0x0000ffc0)|((code>>9)&0x30)|((sprite[0]>>16)&0xf); | ||||
|   } else { | ||||
|     delta<<=4; // Delta of address
 | ||||
|     pal=((code>>9)&0x30)|(sh<<6); | ||||
| 
 | ||||
|     if(sh && (code&0x6000) == 0x6000) { | ||||
|       if(code&0x0800) fTileFunc=TileFlipSH; | ||||
|       else            fTileFunc=TileNormSH; | ||||
|     } else { | ||||
|       if(code&0x0800) fTileFunc=TileFlip; | ||||
|       else            fTileFunc=TileNorm; | ||||
|     } | ||||
| 
 | ||||
|     for (; width; width--,sx+=8,tile+=delta) | ||||
|     { | ||||
|       if(sx<=0)   continue; | ||||
|       if(sx>=328) break; // Offscreen
 | ||||
| 
 | ||||
|       tile&=0x7fff; // Clip tile address
 | ||||
|       fTileFunc(sx,tile,pal); | ||||
|     } | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| void DrawSpritesFromCache(int *hc, int sh) | ||||
| { | ||||
|   int code, tile, sx, delta, width; | ||||
|   int pal; | ||||
|   int (*fTileFunc)(int sx,int addr,int pal); | ||||
| 
 | ||||
|   // *(*hc)++ = (tile<<16)|((code&0x0800)<<5)|((sx<<6)&0x0000ffc0)|((code>>9)&0x30)|((sprite[0]>>24)&0xf);
 | ||||
| 
 | ||||
|   while((code=*hc++)) { | ||||
|     pal=(code&0x30); | ||||
|     delta=code&0xf; | ||||
|     width=delta>>2; delta&=3; | ||||
|     width++; delta++; // Width and height in tiles
 | ||||
|     if (code&0x10000) delta=-delta; // Flip X
 | ||||
|     delta<<=4; | ||||
|     tile=((unsigned int)code>>17)<<1; | ||||
|     sx=(code<<16)>>22; // sx can be negative (start offscreen), so sign extend
 | ||||
| 
 | ||||
|     if(sh && pal == 0x30) { //
 | ||||
|       if(code&0x10000) fTileFunc=TileFlipSH; | ||||
|       else             fTileFunc=TileNormSH; | ||||
|     } else { | ||||
|       if(code&0x10000) fTileFunc=TileFlip; | ||||
|       else             fTileFunc=TileNorm; | ||||
|     } | ||||
| 
 | ||||
|     for (; width; width--,sx+=8,tile+=delta) | ||||
|     { | ||||
|       if(sx<=0)   continue; | ||||
|       if(sx>=328) break; // Offscreen
 | ||||
| 
 | ||||
|       tile&=0x7fff; // Clip tile address
 | ||||
|       fTileFunc(sx,tile,pal); | ||||
|     } | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| void BackFill(int reg7, int sh) | ||||
| { | ||||
|   unsigned int back=0; | ||||
|   unsigned int *pd=NULL,*end=NULL; | ||||
| 
 | ||||
|   // Start with a blank scanline (background colour):
 | ||||
|   back=reg7&0x3f; | ||||
|   back|=sh<<6; | ||||
|   back|=back<<8; | ||||
|   back|=back<<16; | ||||
| 
 | ||||
|   pd= (unsigned int *)(HighCol+8); | ||||
|   end=(unsigned int *)(HighCol+8+320); | ||||
| 
 | ||||
|   do { pd[0]=pd[1]=pd[2]=pd[3]=back; pd+=4; } while (pd<end); | ||||
| } | ||||
| 
 | ||||
| // --------------------------------------------
 | ||||
| 
 | ||||
| extern unsigned short HighPal[0x100]; | ||||
| 
 | ||||
| void FinalizeLineBGR444(int sh) | ||||
| { | ||||
|   unsigned short *pd=DrawLineDest; | ||||
|   unsigned char  *ps=HighCol+8; | ||||
|   unsigned short *pal=Pico.cram; | ||||
|   int len, i, t; | ||||
| 
 | ||||
|   if (Pico.video.reg[12]&1) { | ||||
|     len = 320; | ||||
|   } else { | ||||
|     if(!(PicoOpt&0x100)) pd+=32; | ||||
|     len = 256; | ||||
|   } | ||||
| 
 | ||||
|   if(sh) { | ||||
|     pal=HighPal; | ||||
|     if(Pico.m.dirtyPal) { | ||||
|       blockcpy(pal, Pico.cram, 0x40*2); | ||||
|       // shadowed pixels
 | ||||
|       for(i = 0x3f; i >= 0; i--) | ||||
|         pal[0x40|i] = pal[0xc0|i] = (unsigned short)((pal[i]>>1)&0x0777); | ||||
|       // hilighted pixels
 | ||||
|       for(i = 0x3f; i >= 0; i--) { | ||||
|         t=pal[i]&0xeee;t+=0x444;if(t&0x10)t|=0xe;if(t&0x100)t|=0xe0;if(t&0x1000)t|=0xe00;t&=0xeee; | ||||
|         pal[0x80|i]=(unsigned short)t; | ||||
|       } | ||||
|       Pico.m.dirtyPal = 0; | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   for(i = 0; i < len; i++) | ||||
|     pd[i] = pal[ps[i]]; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| void FinalizeLineRGB555(int sh) | ||||
| { | ||||
|   unsigned short *pd=DrawLineDest; | ||||
|   unsigned char  *ps=HighCol+8; | ||||
|   unsigned short *pal=HighPal; | ||||
|   int len, i, t, dirtyPal = Pico.m.dirtyPal; | ||||
| 
 | ||||
|   if(dirtyPal) { | ||||
|     unsigned short *ppal=Pico.cram; | ||||
|     for(i = 0x3f; i >= 0; i--) | ||||
|       pal[i] = (unsigned short) (((ppal[i]&0x00f)<<12)|((ppal[i]&0x0f0)<<3)|((ppal[i]&0xf00)>>7)); | ||||
|     Pico.m.dirtyPal = 0; | ||||
|   } | ||||
| 
 | ||||
|   if (Pico.video.reg[12]&1) { | ||||
|     len = 320; | ||||
|   } else { | ||||
|     if(!(PicoOpt&0x100)) pd+=32; | ||||
|     len = 256; | ||||
|   } | ||||
| 
 | ||||
|   if(sh) { | ||||
|     if(dirtyPal) { | ||||
|       // shadowed pixels
 | ||||
|       for(i = 0x3f; i >= 0; i--) | ||||
|         pal[0x40|i] = pal[0xc0|i] = (unsigned short)((pal[i]>>1)&0x738e); | ||||
|       // hilighted pixels
 | ||||
|       for(i = 0x3f; i >= 0; i--) { | ||||
|         t=pal[i]&0xe71c;t+=0x4208;if(t&0x20)t|=0x1c;if(t&0x800)t|=0x700;if(t&0x10000)t|=0xe000;t&=0xe71c; | ||||
|         pal[0x80|i]=(unsigned short)t; | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   for(i = 0; i < len; i++) | ||||
|     pd[i] = pal[ps[i]]; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
							
								
								
									
										14
									
								
								psp/emu.c
									
										
									
									
									
								
							
							
						
						
									
										14
									
								
								psp/emu.c
									
										
									
									
									
								
							|  | @ -276,7 +276,7 @@ static void EmuScanPrepare(void) | |||
| 		do_pal_update(1); | ||||
| } | ||||
| 
 | ||||
| static int EmuScanSlow(unsigned int num, void *sdata) | ||||
| static int EmuScanSlow(unsigned int num) | ||||
| { | ||||
| 	if (!(Pico.video.reg[1]&8)) num += 8; | ||||
| 
 | ||||
|  | @ -422,7 +422,7 @@ static void blit2(const char *fps, const char *notice, int lagging_behind) | |||
| 
 | ||||
| 	//dbg_text();
 | ||||
| 
 | ||||
| 	if ((emu_opt & 0x400) && (PicoMCD & 1)) | ||||
| 	if ((emu_opt & 0x400) && (PicoAHW & PAHW_MCD)) | ||||
| 		cd_leds(); | ||||
| 
 | ||||
| 	if (currentConfig.EmuOpt & 0x2000) { // want vsync
 | ||||
|  | @ -467,7 +467,7 @@ static void vidResetMode(void) | |||
| 
 | ||||
| 	// slow rend.
 | ||||
| 	PicoDrawSetColorFormat(-1); | ||||
| 	PicoScan = EmuScanSlow; | ||||
| 	PicoScanEnd = EmuScanSlow; | ||||
| 
 | ||||
| 	localPal[0xe0] = 0; | ||||
| 	Pico.m.dirtyPal = 1; | ||||
|  | @ -670,7 +670,7 @@ void emu_forcedFrame(void) | |||
| 	memset32_uncached((int *)psp_screen + 512*264*2/4, 0, 512*8*2/4); | ||||
| 
 | ||||
| 	PicoDrawSetColorFormat(-1); | ||||
| 	PicoScan = EmuScanSlow; | ||||
| 	PicoScanEnd = EmuScanSlow; | ||||
| 	EmuScanPrepare(); | ||||
| 	PicoFrameDrawOnly(); | ||||
| 	blit1(); | ||||
|  | @ -855,7 +855,7 @@ void emu_Loop(void) | |||
| 	target_frametime = Pico.m.pal ? (1000000<<8)/50 : (1000000<<8)/60+1; | ||||
| 	reset_timing = 1; | ||||
| 
 | ||||
| 	if (PicoMCD & 1) { | ||||
| 	if (PicoAHW & PAHW_MCD) { | ||||
| 		// prepare CD buffer
 | ||||
| 		PicoCDBufferInit(); | ||||
| 		// mp3...
 | ||||
|  | @ -1001,7 +1001,7 @@ void emu_Loop(void) | |||
| 	} | ||||
| 
 | ||||
| 
 | ||||
| 	if (PicoMCD & 1) PicoCDBufferFree(); | ||||
| 	if (PicoAHW & PAHW_MCD) PicoCDBufferFree(); | ||||
| 
 | ||||
| 	if (PsndOut != NULL) { | ||||
| 		PsndOut = NULL; | ||||
|  | @ -1028,7 +1028,7 @@ void emu_ResetGame(void) | |||
| 
 | ||||
| void emu_HandleResume(void) | ||||
| { | ||||
| 	if (!(PicoMCD & 1)) return; | ||||
| 	if (!(PicoAHW & PAHW_MCD)) return; | ||||
| 
 | ||||
| 	// reopen files..
 | ||||
| 	if (Pico_mcd->TOC.Tracks[0].F != NULL) | ||||
|  |  | |||
|  | @ -573,7 +573,7 @@ static void draw_savestate_bg(int slot) | |||
| 	} | ||||
| 
 | ||||
| 	if (file) { | ||||
| 		if (PicoMCD & 1) { | ||||
| 		if (PicoAHW & PAHW_MCD) { | ||||
| 			PicoCdLoadStateGfx(file); | ||||
| 		} else { | ||||
| 			areaSeek(file, 0x10020, SEEK_SET);  // skip header and RAM in state file
 | ||||
|  |  | |||
|  | @ -460,7 +460,7 @@ int mp3_get_offset(void) // 0-1023 | |||
| 	unsigned int offs1024 = 0; | ||||
| 	int cdda_on; | ||||
| 
 | ||||
| 	cdda_on = (PicoMCD & 1) && (PicoOpt&0x800) && !(Pico_mcd->s68k_regs[0x36] & 1) && | ||||
| 	cdda_on = (PicoAHW & PAHW_MCD) && (PicoOpt&0x800) && !(Pico_mcd->s68k_regs[0x36] & 1) && | ||||
| 			(Pico_mcd->scd.Status_CDC & 1) && mp3_handle >= 0; | ||||
| 
 | ||||
| 	if (cdda_on) { | ||||
|  |  | |||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 notaz
						notaz