gw.py 9.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348
  1. # gw.py
  2. #
  3. # Greaseweazle control script.
  4. #
  5. # Written & released by Keir Fraser <keir.xen@gmail.com>
  6. #
  7. # This is free and unencumbered software released into the public domain.
  8. # See the file COPYING for more details, or visit <http://unlicense.org>.
  9. import sys, struct, argparse, serial, collections
  10. from timeit import default_timer as timer
  11. # 40MHz
  12. scp_freq = 40000000
  13. CMD_GET_INFO = 0
  14. CMD_SEEK = 1
  15. CMD_SIDE = 2
  16. CMD_SET_DELAYS = 3
  17. CMD_GET_DELAYS = 4
  18. CMD_MOTOR = 5
  19. CMD_READ_FLUX = 6
  20. CMD_WRITE_FLUX = 7
  21. CMD_GET_FLUX_STATUS = 8
  22. CMD_GET_READ_INFO = 9
  23. ACK_OKAY = 0
  24. ACK_BAD_COMMAND = 1
  25. ACK_NO_INDEX = 2
  26. ACK_NO_TRK0 = 3
  27. ACK_FLUX_OVERFLOW = 4
  28. ACK_FLUX_UNDERFLOW = 5
  29. ACK_WRPROT = 6
  30. ACK_MAX = 6
  31. ack_str = [
  32. "Okay", "Bad Command", "No Index", "Track 0 not found",
  33. "Flux Overflow", "Flux Underflow", "Disk is Write Protected" ]
  34. class CmdError(Exception):
  35. def __init__(self, code):
  36. self.code = code
  37. def __str__(self):
  38. if self.code <= ACK_MAX:
  39. return ack_str[self.code]
  40. return "Unknown Error (%u)" % self.code
  41. def send_cmd(cmd):
  42. ser.write(cmd)
  43. (c,r) = struct.unpack("2B", ser.read(2))
  44. assert c == cmd[0]
  45. if r != 0:
  46. raise CmdError(r)
  47. def get_fw_info():
  48. send_cmd(struct.pack("3B", CMD_GET_INFO, 3, 0))
  49. x = struct.unpack("<4BI24x", ser.read(32))
  50. return x
  51. def print_fw_info(info):
  52. (major, minor, max_revs, max_cmd, freq) = info
  53. print("Greaseweazle v%u.%u" % (major, minor))
  54. print("Max revs %u" % (max_revs))
  55. print("Max cmd %u" % (max_cmd))
  56. print("Sample frequency: %.2f MHz" % (freq / 1000000))
  57. def seek(cyl, side):
  58. send_cmd(struct.pack("3B", CMD_SEEK, 3, cyl))
  59. send_cmd(struct.pack("3B", CMD_SIDE, 3, side))
  60. def get_delays():
  61. send_cmd(struct.pack("2B", CMD_GET_DELAYS, 2))
  62. return struct.unpack("<4H", ser.read(4*2))
  63. def print_delays(x):
  64. (step_delay, seek_settle, motor_delay, auto_off) = x
  65. print("Step Delay: %ums" % step_delay)
  66. print("Settle Time: %ums" % seek_settle)
  67. print("Motor Delay: %ums" % motor_delay)
  68. print("Auto Off: %ums" % auto_off)
  69. def set_delays(step_delay = None, seek_settle = None,
  70. motor_delay = None, auto_off = None):
  71. (_step_delay, _seek_settle, _motor_delay, _auto_off) = get_delays()
  72. if not step_delay: step_delay = _step_delay
  73. if not seek_settle: seek_settle = _seek_settle
  74. if not motor_delay: motor_delay = _motor_delay
  75. if not auto_off: auto_off = _auto_off
  76. send_cmd(struct.pack("<2B4H", CMD_SET_DELAYS, 10,
  77. step_delay, seek_settle, motor_delay, auto_off))
  78. def motor(state):
  79. send_cmd(struct.pack("3B", CMD_MOTOR, 3, int(state)))
  80. def get_read_info():
  81. send_cmd(struct.pack("2B", CMD_GET_READ_INFO, 2))
  82. x = []
  83. for i in range(7):
  84. x.append(struct.unpack("<2I", ser.read(2*4)))
  85. return x
  86. def print_read_info(info):
  87. for (time, samples) in info:
  88. print("%u ticks, %u samples" % (time, samples))
  89. def write_flux(flux):
  90. start = timer()
  91. x = bytearray()
  92. for val in flux:
  93. if val == 0:
  94. pass
  95. elif val < 250:
  96. x.append(val)
  97. else:
  98. high = val // 250
  99. if high <= 5:
  100. x.append(249+high)
  101. x.append(1 + val%250)
  102. else:
  103. x.append(255)
  104. x.append(1 | (val<<1) & 255)
  105. x.append(1 | (val>>6) & 255)
  106. x.append(1 | (val>>13) & 255)
  107. x.append(1 | (val>>20) & 255)
  108. x.append(0) # End of Stream
  109. end = timer()
  110. #print("%u flux -> %u bytes in %f seconds" % (len(flux), len(x), end-start))
  111. retry = 0
  112. while True:
  113. start = timer()
  114. send_cmd(struct.pack("2B", CMD_WRITE_FLUX, 2))
  115. ser.write(x)
  116. ser.read(1) # Sync with Greaseweazle
  117. try:
  118. send_cmd(struct.pack("2B", CMD_GET_FLUX_STATUS, 2))
  119. except CmdError as error:
  120. if error.code == ACK_FLUX_UNDERFLOW and retry < 5:
  121. retry += 1
  122. print("Retry #%u..." % retry)
  123. continue;
  124. raise
  125. end = timer()
  126. #print("Track written in %f seconds" % (end-start))
  127. break
  128. def read_flux(nr_revs):
  129. retry = 0
  130. while True:
  131. start = timer()
  132. x = collections.deque()
  133. send_cmd(struct.pack("3B", CMD_READ_FLUX, 3, nr_revs))
  134. nr = 0
  135. while True:
  136. x += ser.read(1)
  137. x += ser.read(ser.in_waiting)
  138. nr += 1;
  139. if x[-1] == 0:
  140. break
  141. try:
  142. send_cmd(struct.pack("2B", CMD_GET_FLUX_STATUS, 2))
  143. except CmdError as error:
  144. if error.code == ACK_FLUX_OVERFLOW and retry < 5:
  145. retry += 1
  146. print("Retry #%u..." % retry)
  147. del x
  148. continue;
  149. raise
  150. end = timer()
  151. break
  152. #print("Read %u bytes in %u batches in %f seconds" % (len(x), nr, end-start))
  153. start = timer()
  154. y = []
  155. while x:
  156. i = x.popleft()
  157. if i < 250:
  158. y.append(i)
  159. elif i == 255:
  160. val = (x.popleft() & 254) >> 1
  161. val += (x.popleft() & 254) << 6
  162. val += (x.popleft() & 254) << 13
  163. val += (x.popleft() & 254) << 20
  164. y.append(val)
  165. else:
  166. val = (i - 249) * 250
  167. val += x.popleft() - 1
  168. y.append(val)
  169. assert y[-1] == 0
  170. y = y[:-1]
  171. end = timer()
  172. #print("Processed %u flux values in %f seconds" % (len(y), end-start))
  173. return y
  174. def read(args):
  175. factor = scp_freq / sample_freq
  176. trk_dat = bytearray()
  177. trk_offs = []
  178. if args.single_sided:
  179. track_range = range(args.scyl, args.ecyl+1)
  180. nr_sides = 1
  181. else:
  182. track_range = range(args.scyl*2, (args.ecyl+1)*2)
  183. nr_sides = 2
  184. for i in track_range:
  185. cyl = i >> (nr_sides - 1)
  186. side = i & (nr_sides - 1)
  187. print("\rReading Track %u.%u..." % (cyl, side), end="")
  188. trk_offs.append(len(trk_dat))
  189. seek(cyl, side)
  190. flux = read_flux(args.revs)
  191. info = get_read_info()[:args.revs]
  192. #print_read_info(info)
  193. trk_dat += struct.pack("<3sB", b"TRK", i)
  194. dat_off = 4 + args.revs*12
  195. for (time, samples) in info:
  196. time = int(round(time * factor))
  197. trk_dat += struct.pack("<III", time, samples, dat_off)
  198. dat_off += samples * 2
  199. rem = 0.0
  200. for x in flux:
  201. y = x * factor + rem
  202. val = int(round(y))
  203. rem = y - val
  204. while val >= 65536:
  205. trk_dat.append(0)
  206. trk_dat.append(0)
  207. val -= 65536
  208. if val == 0:
  209. val = 1
  210. trk_dat.append(val>>8)
  211. trk_dat.append(val&255)
  212. print()
  213. csum = 0
  214. for x in trk_dat:
  215. csum += x
  216. trk_offs_dat = bytearray()
  217. for x in trk_offs:
  218. trk_offs_dat += struct.pack("<I", 0x2b0 + x)
  219. trk_offs_dat += bytes(0x2a0 - len(trk_offs_dat))
  220. for x in trk_offs_dat:
  221. csum += x
  222. ds_flag = 0
  223. if args.single_sided:
  224. ds_flag = 1
  225. header_dat = struct.pack("<3s9BI",
  226. b"SCP", # Signature
  227. 0, # Version
  228. 0x80, # DiskType = Other
  229. args.revs, # Nr Revolutions
  230. args.scyl, # Start track
  231. args.ecyl, # End track
  232. 0x21, # Flags = Index, Footer
  233. 0, # 16-bit cell width
  234. ds_flag, # Double Sided
  235. 0, # 25ns capture
  236. csum & 0xffffffff)
  237. with open(args.file, "wb") as f:
  238. f.write(header_dat)
  239. f.write(trk_offs_dat)
  240. f.write(trk_dat)
  241. def write(args):
  242. factor = sample_freq / scp_freq
  243. with open(args.file, "rb") as f:
  244. dat = f.read()
  245. header = struct.unpack("<3s9BI", dat[0:16])
  246. assert header[0] == b"SCP"
  247. trk_offs = struct.unpack("<168I", dat[16:0x2b0])
  248. if args.single_sided:
  249. track_range = range(args.scyl, args.ecyl+1)
  250. nr_sides = 1
  251. else:
  252. track_range = range(args.scyl*2, (args.ecyl+1)*2)
  253. nr_sides = 2
  254. for i in track_range:
  255. cyl = i >> (nr_sides - 1)
  256. side = i & (nr_sides - 1)
  257. print("\rWriting Track %u.%u..." % (cyl, side), end="")
  258. if trk_offs[i] == 0:
  259. continue
  260. seek(cyl, side)
  261. thdr = struct.unpack("<3sBIII", dat[trk_offs[i]:trk_offs[i]+16])
  262. (sig,_,_,samples,off) = thdr
  263. assert sig == b"TRK"
  264. tdat = dat[trk_offs[i]+off:trk_offs[i]+off+samples*2]
  265. flux = []
  266. rem = 0.0
  267. for i in range(0,len(tdat),2):
  268. x = tdat[i]*256 + tdat[i+1]
  269. if x == 0:
  270. rem += 65536.0
  271. continue
  272. y = x * factor + rem
  273. val = int(round(y))
  274. rem = y - val
  275. flux.append(val)
  276. write_flux(flux)
  277. print()
  278. def main(argv):
  279. actions = {
  280. "read" : read,
  281. "write" : write
  282. }
  283. parser = argparse.ArgumentParser(
  284. formatter_class=argparse.ArgumentDefaultsHelpFormatter)
  285. parser.add_argument("action")
  286. parser.add_argument("--revs", type=int, default=3,
  287. help="number of revolutions to read per track")
  288. parser.add_argument("--scyl", type=int, default=0,
  289. help="first cylinder to read/write")
  290. parser.add_argument("--ecyl", type=int, default=81,
  291. help="last cylinder to read/write")
  292. parser.add_argument("--single-sided", action="store_true")
  293. # parser.add_argument("--total", type=float, default=8.0,
  294. # help="total length, seconds")
  295. parser.add_argument("file", help="in/out filename")
  296. parser.add_argument("device", help="serial device")
  297. args = parser.parse_args(argv[1:])
  298. global ser
  299. ser = serial.Serial(args.device)
  300. ser.send_break()
  301. ser.reset_input_buffer()
  302. global sample_freq
  303. info = get_fw_info()
  304. #print_fw_info(info)
  305. sample_freq = info[4]
  306. #set_delays(step_delay=3)
  307. #print_delays(get_delays())
  308. actions[args.action](args)
  309. motor(False)
  310. if __name__ == "__main__":
  311. try:
  312. main(sys.argv)
  313. except CmdError as error:
  314. print("Command Failed: %s" % error)
  315. motor(False)