Advertisement
mcleod_ideafix

Jupiter ACE TAP to TZX converter

Nov 17th, 2018
543
0
Never
Not a member of Pastebin yet? Sign Up, it unlocks many cool features!
C 4.66 KB | None | 0 0
  1. /*
  2. ACETAP2TZX
  3. ----------
  4.  
  5. (C)2011 Miguel Angel Rodriguez Jodar (McLeod/IdeaFix).
  6.  
  7. A quick and dirty utility to convert TAP files designed for the Jupiter ACE
  8. computer, into more flexible TZX files.
  9. This utility outputs a TZX confirming to version 1.20
  10.  
  11. This program is free software: you can redistribute it and/or modify it
  12. under the terms of the GNU General Public License as published by the Free
  13. Software Foundation, either version 3 of the License, or (at your option)
  14. any later version.
  15.  
  16. This program is distributed in the hope that it will be useful, but WITHOUT
  17. ANY WARRANTY; without even the implied warranty of  MERCHANTABILITY or
  18. FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
  19. more details.
  20.  
  21. You should have received a copy of the GNU General Public License along with
  22. this program.  If not, see <http://www.gnu.org/licenses/>.
  23.  
  24. Changelog
  25. ---------
  26. 2011-01-04:
  27. Simplified TZX enconding using just ID11 and ID13.
  28.  
  29.  
  30. 2011-01-01:
  31. First version released to the public. For each block of TAP data, the following
  32. is written to the TZX output:
  33.     A pure tone (pilot), two pulses (sync), data, and a single 0 bit.
  34.  
  35. Multiple TAP blocks are treated as pairs, assuming that each pair contains
  36. a header block (26 bytes) and a data block.
  37.  
  38. Possibly multiple security flaws in the code, and a number of unknown errors.
  39. The code has been written and tested in less than 1 hour...
  40. */
  41.  
  42.  
  43. #include <stdio.h>
  44. #include <stdlib.h>
  45. #include <string.h>
  46.  
  47. /*
  48. These timings are adjusted from the original timings from the 3.25MHz
  49. Jupiter ACE to a Z80 frequency of 3.5MHz, as required by the TZX file format
  50. standard (1.20). To get the actual Jupiter ACE timings, multiply each
  51. value by 325/350. More information about these timings at:
  52. http://jupiterace.proboards.com/index.cgi?board=programmingaceforth&action=display&thread=266
  53. */
  54.  
  55. #define PILOT 2114
  56. #define SYNC1 602
  57. #define SYNC2 826
  58. #define ZERO 812
  59. #define ONE 1666
  60. #define ENDMARK1 972
  61. #define ENDMARK2 4509
  62.  
  63. void WriteTZXBlock (FILE *, unsigned char, unsigned char *, int);
  64. int  ReadTAPBlock (FILE *, unsigned char *);
  65. void WriteTZXHeader (FILE *);
  66.  
  67. int main (int argc, char *argv[])
  68. {
  69.     FILE *fi, *fo;
  70.     char ni[256],no[256];
  71.     unsigned char ffhd=0;
  72.     unsigned char *bloque;
  73.     int lbloque;
  74.    
  75.     if (argc<2)
  76.     {
  77.         printf ("ERROR! Need a file to convert!\n");
  78.         exit(-1);
  79.     }
  80.    
  81.     strncpy (ni, argv[1], 255);
  82.     ni[255]=0;
  83.    
  84.     strcpy (no, ni);
  85.     strcat (no, ".tzx");
  86.    
  87.     bloque=(unsigned char *)malloc(65536);
  88.     if (!bloque)
  89.     {
  90.         printf ("ERROR! Out of memory (shouldn't happen!)\n");
  91.         exit(-1);
  92.     }
  93.  
  94.     fi=fopen(ni,"rb");
  95.     if (!fi)
  96.     {
  97.         printf ("ERROR! Cannot open '%s'\n", ni);
  98.         exit(-1);
  99.     }
  100.    
  101.     fo=fopen(no,"wb");
  102.     if (!fo)
  103.     {
  104.         printf ("ERROR! Cannot create '%s'\n", no);
  105.         exit(-1);
  106.     }
  107.    
  108.     WriteTZXHeader (fo);
  109.    
  110.     lbloque = ReadTAPBlock (fi, bloque);
  111.     while(lbloque>0)
  112.     {
  113.         WriteTZXBlock (fo, ffhd, bloque, lbloque);
  114.         ffhd=~ffhd;
  115.         lbloque = ReadTAPBlock (fi, bloque);
  116.     }
  117.    
  118.     fclose(fi);
  119.     fclose(fo);
  120.     free(bloque);
  121.    
  122.     return 0;
  123. }
  124.  
  125.  
  126. void WriteTZXBlock (FILE *fo, unsigned char ffhd, unsigned char *bloque, int lbloque)
  127. {
  128.     unsigned char id2b[5]={1,0,0,0,1};
  129.     unsigned char id13[6]={0x13, 2, ENDMARK1&0xff, (ENDMARK1>>8)&0xff, ENDMARK2&0xff, (ENDMARK2>>8)&0xff};
  130.     unsigned char id11[19]={0x11, PILOT&0xff, (PILOT>>8)&0xff,
  131.                                  SYNC1&0xff, (SYNC1>>8)&0xff,
  132.                                  SYNC2&0xff, (SYNC2>>8)&0xff,
  133.                                  ZERO&0xff, (ZERO>>8)&0xff,
  134.                                  ONE&0xff, (ONE>>8)&0xff,
  135.                                  0, 0, /* length of pilot tone */
  136.                                  8, 0, 0,
  137.                                  0, 0, 0 /* length of data to follow */
  138.                           };
  139.                          
  140.    
  141.     id11[12]=(!ffhd)? 0x20 : 0x04;  /* length of pilot tone is different for header and data blocks */
  142.     id11[16]=(lbloque+1)&0xff;      /* actual data in the TZX file has one byte added: the flag byte */
  143.     id11[17]=((lbloque+1)>>8)&0xff;  
  144.    
  145.     /* fwrite (id2b, 1, 5, fo);  set signal level */
  146.     fwrite (id11, 1, 19, fo); /* turbo data block */
  147.     fwrite (&ffhd, 1, 1, fo); /* flag byte */
  148.     fwrite (bloque, 1, lbloque, fo); /* the actual data */
  149.     fwrite (id13, 1, 6, fo); /* end mark */
  150. }
  151.  
  152.  
  153. int ReadTAPBlock (FILE *fi, unsigned char *bloque)
  154. {
  155.     int leido;
  156.     int lbloque;
  157.     unsigned char lowb, hib;
  158.    
  159.     leido=fread (&lowb, 1, 1, fi);
  160.     leido=fread (&hib, 1, 1, fi);
  161.     if (leido<1)
  162.         return 0;
  163.    
  164.     lbloque=(hib<<8) | lowb;
  165.     leido=fread (bloque, 1, lbloque, fi);
  166.     if (leido!=lbloque)
  167.         return 0;
  168.        
  169.     return lbloque;
  170. }
  171.  
  172.  
  173. void WriteTZXHeader (FILE *fo)
  174. {
  175.     unsigned char headtzx[10]={'Z','X','T','a','p','e','!',0x1a, 1, 13};
  176.    
  177.     fwrite (headtzx, 1, 10, fo);
  178. }
Advertisement
Add Comment
Please, Sign In to add comment
Advertisement