Save_PRG() supports C64 hires bitmap mode
This commit is contained in:
		
							parent
							
								
									c820ba61ac
								
							
						
					
					
						commit
						19dec01d4f
					
				@ -1161,15 +1161,17 @@ static int Save_C64_window(enum c64_format *saveFormat, byte *saveWhat, word *lo
 | 
				
			|||||||
  return button==1;
 | 
					  return button==1;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
/// Save a C64 hires picture
 | 
					
 | 
				
			||||||
///
 | 
					/**
 | 
				
			||||||
/// c64 hires is 320x200 with only 2 colors per 8x8 block.
 | 
					 * Encode a C64 HiRes Bitmap picture.
 | 
				
			||||||
static int Save_C64_hires(T_IO_Context *context, byte saveWhat, word loadAddr)
 | 
					 * 320x200 pixels, with only 2 different colors per 8x8 block.
 | 
				
			||||||
 | 
					 *
 | 
				
			||||||
 | 
					 * 8000 bytes bitmap, 1000 bytes screen RAM
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					static int Encode_C64_hires(T_IO_Context * context, byte * bitmap, byte * screen_ram)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
  int i, pos = 0;
 | 
					  int i, pos = 0;
 | 
				
			||||||
  word cx, cy, x, y;
 | 
					  word cx, cy, x, y;
 | 
				
			||||||
  byte screen_ram[1000],bitmap[8000];
 | 
					 | 
				
			||||||
  FILE *file;
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
  for(cy=0; cy<25; cy++) // Character line, 25 lines
 | 
					  for(cy=0; cy<25; cy++) // Character line, 25 lines
 | 
				
			||||||
  {
 | 
					  {
 | 
				
			||||||
@ -1246,7 +1248,21 @@ static int Save_C64_hires(T_IO_Context *context, byte saveWhat, word loadAddr)
 | 
				
			|||||||
      }
 | 
					      }
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
 | 
					  return 0;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/// Save a C64 hires picture
 | 
				
			||||||
 | 
					///
 | 
				
			||||||
 | 
					/// c64 hires is 320x200 with only 2 colors per 8x8 block.
 | 
				
			||||||
 | 
					static int Save_C64_hires(T_IO_Context *context, byte saveWhat, word loadAddr)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
					  byte screen_ram[1000],bitmap[8000];
 | 
				
			||||||
 | 
					  FILE *file;
 | 
				
			||||||
 | 
					  int ret;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  ret = Encode_C64_hires(context, bitmap, screen_ram);
 | 
				
			||||||
 | 
					  if (ret != 0)
 | 
				
			||||||
 | 
					    return ret;
 | 
				
			||||||
  file = Open_file_write(context);
 | 
					  file = Open_file_write(context);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  if(!file)
 | 
					  if(!file)
 | 
				
			||||||
@ -1722,7 +1738,17 @@ void Save_PRG(T_IO_Context * context)
 | 
				
			|||||||
  saveFormat = (context->Width == 320) ? F_hires : F_multi;
 | 
					  saveFormat = (context->Width == 320) ? F_hires : F_multi;
 | 
				
			||||||
  File_error = 1;
 | 
					  File_error = 1;
 | 
				
			||||||
  Set_saving_layer(context, 0);
 | 
					  Set_saving_layer(context, 0);
 | 
				
			||||||
  File_error = Encode_C64_multicolor(context, bitmap, screen_ram, color_ram, &background);
 | 
					  switch (saveFormat)
 | 
				
			||||||
 | 
					  {
 | 
				
			||||||
 | 
					    case F_hires:
 | 
				
			||||||
 | 
					      File_error = Encode_C64_hires(context, bitmap, screen_ram);
 | 
				
			||||||
 | 
					      break;
 | 
				
			||||||
 | 
					    case F_multi:
 | 
				
			||||||
 | 
					      File_error = Encode_C64_multicolor(context, bitmap, screen_ram, color_ram, &background);
 | 
				
			||||||
 | 
					      break;
 | 
				
			||||||
 | 
					    default:
 | 
				
			||||||
 | 
					      GFX2_Log(GFX2_ERROR, "Save_PRG(): format %d not handled (yet?)\n", saveFormat);
 | 
				
			||||||
 | 
					  }
 | 
				
			||||||
  if (File_error == 0)
 | 
					  if (File_error == 0)
 | 
				
			||||||
  {
 | 
					  {
 | 
				
			||||||
    FILE *file;
 | 
					    FILE *file;
 | 
				
			||||||
@ -1736,7 +1762,7 @@ void Save_PRG(T_IO_Context * context)
 | 
				
			|||||||
    }
 | 
					    }
 | 
				
			||||||
    if (!Write_bytes(file, picview_prg, sizeof(picview_prg)))
 | 
					    if (!Write_bytes(file, picview_prg, sizeof(picview_prg)))
 | 
				
			||||||
      File_error = 2;
 | 
					      File_error = 2;
 | 
				
			||||||
    Write_byte(file, 0x30); // Mode : Bitmap + Multicolor
 | 
					    Write_byte(file, saveFormat == F_multi ? 0x30 : 0x20); // Mode : 0x40 Extended bg, 0x20 Bitmap, 0x10 Multicolor
 | 
				
			||||||
    n = PackBits_pack_buffer(NULL, bitmap, 8000);
 | 
					    n = PackBits_pack_buffer(NULL, bitmap, 8000);
 | 
				
			||||||
    GFX2_Log(GFX2_DEBUG, "PackBits of bitmap : 8000 => %d bytes\n", n + 1);
 | 
					    GFX2_Log(GFX2_DEBUG, "PackBits of bitmap : 8000 => %d bytes\n", n + 1);
 | 
				
			||||||
    if (n >= 0 && n < 7999)
 | 
					    if (n >= 0 && n < 7999)
 | 
				
			||||||
@ -1764,15 +1790,19 @@ void Save_PRG(T_IO_Context * context)
 | 
				
			|||||||
      Write_byte(file, 0x20); // screen RAM / no packing
 | 
					      Write_byte(file, 0x20); // screen RAM / no packing
 | 
				
			||||||
      Write_bytes(file, screen_ram, 1000);
 | 
					      Write_bytes(file, screen_ram, 1000);
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
    //Write_byte(file, 0x30); // color RAM / no packing
 | 
					    if (saveFormat == F_multi)
 | 
				
			||||||
    //Write_bytes(file, color_ram, 1000);
 | 
					    {
 | 
				
			||||||
    Write_byte(file, 0x32); // color RAM / special color RAM packing
 | 
					      //Write_byte(file, 0x30); // color RAM / no packing
 | 
				
			||||||
    n = C64_color_ram_pack(file, color_ram, 1000);
 | 
					      //Write_bytes(file, color_ram, 1000);
 | 
				
			||||||
    if (n < 0)
 | 
					      Write_byte(file, 0x32); // color RAM / special color RAM packing
 | 
				
			||||||
      File_error = 1;
 | 
					      n = C64_color_ram_pack(file, color_ram, 1000);
 | 
				
			||||||
    GFX2_Log(GFX2_DEBUG, "custom packing of color RAM : 1000 => %d bytes\n", n);
 | 
					      if (n < 0)
 | 
				
			||||||
    Write_byte(file, 0x42); // border/background/etc. / color ram RLE packing
 | 
					        File_error = 1;
 | 
				
			||||||
    Write_byte(file, background | 0x10);
 | 
					      GFX2_Log(GFX2_DEBUG, "custom packing of color RAM : 1000 => %d bytes\n", n);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      Write_byte(file, 0x42); // border/background/etc. / color ram RLE packing
 | 
				
			||||||
 | 
					      Write_byte(file, background | 0x10);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
    Write_byte(file, 0); // end of file
 | 
					    Write_byte(file, 0); // end of file
 | 
				
			||||||
    fclose(file);
 | 
					    fclose(file);
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
 | 
				
			|||||||
@ -404,11 +404,12 @@ ret:
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
int Test_C64_Formats(void)
 | 
					int Test_C64_Formats(void)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
  int i;
 | 
					  int i, j;
 | 
				
			||||||
  int ok = 0;
 | 
					  int ok = 0;
 | 
				
			||||||
  T_IO_Context context;
 | 
					  T_IO_Context context;
 | 
				
			||||||
  char path[256];
 | 
					  char path[256];
 | 
				
			||||||
  T_GFX2_Surface * testpicmulti = NULL;
 | 
					  T_GFX2_Surface * testpicmulti = NULL;
 | 
				
			||||||
 | 
					  T_GFX2_Surface * testpichires = NULL;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  memset(&context, 0, sizeof(context));
 | 
					  memset(&context, 0, sizeof(context));
 | 
				
			||||||
  context.Type = CONTEXT_SURFACE;
 | 
					  context.Type = CONTEXT_SURFACE;
 | 
				
			||||||
@ -424,6 +425,16 @@ int Test_C64_Formats(void)
 | 
				
			|||||||
  }
 | 
					  }
 | 
				
			||||||
  testpicmulti = context.Surface;
 | 
					  testpicmulti = context.Surface;
 | 
				
			||||||
  context.Surface = NULL;
 | 
					  context.Surface = NULL;
 | 
				
			||||||
 | 
					  // Load a hires picture
 | 
				
			||||||
 | 
					  context_set_file_path(&context, "../tests/pic-samples/c64/hires/midear.dd");
 | 
				
			||||||
 | 
					  Load_C64(&context);
 | 
				
			||||||
 | 
					  if (File_error != 0)
 | 
				
			||||||
 | 
					  {
 | 
				
			||||||
 | 
					    fprintf(stderr, "Failed to load reference hires picture\n");
 | 
				
			||||||
 | 
					    goto ret;
 | 
				
			||||||
 | 
					  }
 | 
				
			||||||
 | 
					  testpichires = context.Surface;
 | 
				
			||||||
 | 
					  context.Surface = NULL;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  ok = 1;
 | 
					  ok = 1;
 | 
				
			||||||
  for (i = 0; ok && formats[i].name != NULL; i++)
 | 
					  for (i = 0; ok && formats[i].name != NULL; i++)
 | 
				
			||||||
@ -433,70 +444,86 @@ int Test_C64_Formats(void)
 | 
				
			|||||||
    if (!(formats[i].flags & FLAG_C64))
 | 
					    if (!(formats[i].flags & FLAG_C64))
 | 
				
			||||||
      continue;
 | 
					      continue;
 | 
				
			||||||
    GFX2_Log(GFX2_DEBUG, "Testing format %s (Save)\n", formats[i].name);
 | 
					    GFX2_Log(GFX2_DEBUG, "Testing format %s (Save)\n", formats[i].name);
 | 
				
			||||||
    snprintf(path, sizeof(path), "/tmp/%s.%s", "test", formats[i].name);
 | 
					    for (j = 0; j < 2; j++)
 | 
				
			||||||
    context_set_file_path(&context, path);
 | 
					    {
 | 
				
			||||||
 | 
					      T_GFX2_Surface * ref;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    // save the reference picture
 | 
					      if (j == 0)
 | 
				
			||||||
    context.Surface = testpicmulti;
 | 
					 | 
				
			||||||
    context.Target_address = context.Surface->pixels;
 | 
					 | 
				
			||||||
    context.Pitch = context.Surface->w;
 | 
					 | 
				
			||||||
    context.Width = context.Surface->w;
 | 
					 | 
				
			||||||
    context.Height = context.Surface->h;
 | 
					 | 
				
			||||||
    context.Ratio = PIXEL_WIDE;
 | 
					 | 
				
			||||||
    memcpy(context.Palette, context.Surface->palette, sizeof(T_Palette));
 | 
					 | 
				
			||||||
    context.Format = formats[i].format;
 | 
					 | 
				
			||||||
    File_error = 0;
 | 
					 | 
				
			||||||
    formats[i].Save(&context);
 | 
					 | 
				
			||||||
    context.Surface = NULL;
 | 
					 | 
				
			||||||
    if (File_error != 0)
 | 
					 | 
				
			||||||
    {
 | 
					 | 
				
			||||||
      GFX2_Log(GFX2_ERROR, "Save_%s failed.\n", formats[i].name);
 | 
					 | 
				
			||||||
      ok = 0;
 | 
					 | 
				
			||||||
    }
 | 
					 | 
				
			||||||
    else
 | 
					 | 
				
			||||||
    {
 | 
					 | 
				
			||||||
      FILE * f;
 | 
					 | 
				
			||||||
      // Test the saved file
 | 
					 | 
				
			||||||
      f = fopen(path, "rb");
 | 
					 | 
				
			||||||
      if (f == NULL)
 | 
					 | 
				
			||||||
      {
 | 
					      {
 | 
				
			||||||
        GFX2_Log(GFX2_ERROR, "error opening %s\n", path);
 | 
					        ref = testpicmulti;
 | 
				
			||||||
 | 
					        context.Ratio = PIXEL_WIDE;
 | 
				
			||||||
 | 
					      }
 | 
				
			||||||
 | 
					      else
 | 
				
			||||||
 | 
					      {
 | 
				
			||||||
 | 
					        ref = testpichires;
 | 
				
			||||||
 | 
					        context.Ratio = PIXEL_SIMPLE;
 | 
				
			||||||
 | 
					      }
 | 
				
			||||||
 | 
					      snprintf(path, sizeof(path), "/tmp/%s-%d.%s", "test", j, formats[i].name);
 | 
				
			||||||
 | 
					      context_set_file_path(&context, path);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      // save the reference picture
 | 
				
			||||||
 | 
					      context.Surface = ref;
 | 
				
			||||||
 | 
					      context.Target_address = context.Surface->pixels;
 | 
				
			||||||
 | 
					      context.Pitch = context.Surface->w;
 | 
				
			||||||
 | 
					      context.Width = context.Surface->w;
 | 
				
			||||||
 | 
					      context.Height = context.Surface->h;
 | 
				
			||||||
 | 
					      memcpy(context.Palette, context.Surface->palette, sizeof(T_Palette));
 | 
				
			||||||
 | 
					      context.Format = formats[i].format;
 | 
				
			||||||
 | 
					      File_error = 0;
 | 
				
			||||||
 | 
					      formats[i].Save(&context);
 | 
				
			||||||
 | 
					      context.Surface = NULL;
 | 
				
			||||||
 | 
					      if (File_error != 0)
 | 
				
			||||||
 | 
					      {
 | 
				
			||||||
 | 
					        GFX2_Log(GFX2_ERROR, "Save_%s failed.\n", formats[i].name);
 | 
				
			||||||
        ok = 0;
 | 
					        ok = 0;
 | 
				
			||||||
      }
 | 
					      }
 | 
				
			||||||
      else
 | 
					      else
 | 
				
			||||||
      {
 | 
					      {
 | 
				
			||||||
        File_error = 1;
 | 
					        FILE * f;
 | 
				
			||||||
        formats[i].Test(&context, f);
 | 
					        // Test the saved file
 | 
				
			||||||
        fclose(f);
 | 
					        f = fopen(path, "rb");
 | 
				
			||||||
        if (File_error != 0)
 | 
					        if (f == NULL)
 | 
				
			||||||
        {
 | 
					        {
 | 
				
			||||||
          GFX2_Log(GFX2_ERROR, "Test_%s failed for file %s\n", formats[i].name, path);
 | 
					          GFX2_Log(GFX2_ERROR, "error opening %s\n", path);
 | 
				
			||||||
          ok = 0;
 | 
					          ok = 0;
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
      }
 | 
					        else
 | 
				
			||||||
      memset(context.Palette, -1, sizeof(T_Palette));
 | 
					 | 
				
			||||||
      // load the saved file
 | 
					 | 
				
			||||||
      formats[i].Load(&context);
 | 
					 | 
				
			||||||
      if (File_error != 0 || context.Surface == NULL)
 | 
					 | 
				
			||||||
      {
 | 
					 | 
				
			||||||
        GFX2_Log(GFX2_ERROR, "Load_%s failed for file %s\n", formats[i].name, path);
 | 
					 | 
				
			||||||
        ok = 0;
 | 
					 | 
				
			||||||
      }
 | 
					 | 
				
			||||||
      else
 | 
					 | 
				
			||||||
      {
 | 
					 | 
				
			||||||
        if (memcmp(testpicmulti->pixels, context.Surface->pixels, 160*200) != 0)
 | 
					 | 
				
			||||||
        {
 | 
					        {
 | 
				
			||||||
          GFX2_Log(GFX2_ERROR, "Save_%s/Load_%s: Pixels mismatch\n", formats[i].name, formats[i].name);
 | 
					          File_error = 1;
 | 
				
			||||||
 | 
					          formats[i].Test(&context, f);
 | 
				
			||||||
 | 
					          fclose(f);
 | 
				
			||||||
 | 
					          if (File_error != 0)
 | 
				
			||||||
 | 
					          {
 | 
				
			||||||
 | 
					            GFX2_Log(GFX2_ERROR, "Test_%s failed for file %s\n", formats[i].name, path);
 | 
				
			||||||
 | 
					            ok = 0;
 | 
				
			||||||
 | 
					          }
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        memset(context.Palette, -1, sizeof(T_Palette));
 | 
				
			||||||
 | 
					        // load the saved file
 | 
				
			||||||
 | 
					        formats[i].Load(&context);
 | 
				
			||||||
 | 
					        if (File_error != 0 || context.Surface == NULL)
 | 
				
			||||||
 | 
					        {
 | 
				
			||||||
 | 
					          GFX2_Log(GFX2_ERROR, "Load_%s failed for file %s\n", formats[i].name, path);
 | 
				
			||||||
          ok = 0;
 | 
					          ok = 0;
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
        Free_GFX2_Surface(context.Surface);
 | 
					        else
 | 
				
			||||||
        context.Surface = NULL;
 | 
					        {
 | 
				
			||||||
 | 
					          if (memcmp(ref->pixels, context.Surface->pixels, ref->w * ref->h) != 0)
 | 
				
			||||||
 | 
					          {
 | 
				
			||||||
 | 
					            GFX2_Log(GFX2_ERROR, "Save_%s/Load_%s: Pixels mismatch\n", formats[i].name, formats[i].name);
 | 
				
			||||||
 | 
					            ok = 0;
 | 
				
			||||||
 | 
					          }
 | 
				
			||||||
 | 
					          Free_GFX2_Surface(context.Surface);
 | 
				
			||||||
 | 
					          context.Surface = NULL;
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
      }
 | 
					      }
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
ret:
 | 
					ret:
 | 
				
			||||||
  if (testpicmulti)
 | 
					  if (testpicmulti)
 | 
				
			||||||
    Free_GFX2_Surface(testpicmulti);
 | 
					    Free_GFX2_Surface(testpicmulti);
 | 
				
			||||||
 | 
					  if (testpichires)
 | 
				
			||||||
 | 
					    Free_GFX2_Surface(testpichires);
 | 
				
			||||||
  free(context.File_name);
 | 
					  free(context.File_name);
 | 
				
			||||||
  free(context.File_directory);
 | 
					  free(context.File_directory);
 | 
				
			||||||
  return ok;
 | 
					  return ok;
 | 
				
			||||||
 | 
				
			|||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user