usb.py 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515
  1. # greaseweazle/usb.py
  2. #
  3. # Written & released by Keir Fraser <keir.xen@gmail.com>
  4. #
  5. # This is free and unencumbered software released into the public domain.
  6. # See the file COPYING for more details, or visit <http://unlicense.org>.
  7. import struct
  8. from greaseweazle import version
  9. from greaseweazle import error
  10. from greaseweazle.flux import Flux
  11. ## Control-Path command set
  12. class ControlCmd:
  13. ClearComms = 10000
  14. Normal = 9600
  15. ## Command set
  16. class Cmd:
  17. GetInfo = 0
  18. Update = 1
  19. Seek = 2
  20. Side = 3
  21. SetParams = 4
  22. GetParams = 5
  23. Motor = 6
  24. ReadFlux = 7
  25. WriteFlux = 8
  26. GetFluxStatus = 9
  27. GetIndexTimes = 10
  28. SwitchFwMode = 11
  29. Select = 12
  30. Deselect = 13
  31. SetBusType = 14
  32. SetPin = 15
  33. Reset = 16
  34. EraseFlux = 17
  35. SourceBytes = 18
  36. SinkBytes = 19
  37. str = {
  38. GetInfo: "GetInfo",
  39. Update: "Update",
  40. Seek: "Seek",
  41. Side: "Side",
  42. SetParams: "SetParams",
  43. GetParams: "GetParams",
  44. Motor: "Motor",
  45. ReadFlux: "ReadFlux",
  46. WriteFlux: "WriteFlux",
  47. GetFluxStatus: "GetFluxStatus",
  48. GetIndexTimes: "GetIndexTimes",
  49. SwitchFwMode: "SwitchFwMode",
  50. Select: "Select",
  51. Deselect: "Deselect",
  52. SetBusType: "SetBusType",
  53. SetPin: "SetPin",
  54. Reset: "Reset",
  55. EraseFlux: "EraseFlux",
  56. SourceBytes: "SourceBytes",
  57. SinkBytes: "SinkBytes"
  58. }
  59. ## Command responses/acknowledgements
  60. class Ack:
  61. Okay = 0
  62. BadCommand = 1
  63. NoIndex = 2
  64. NoTrk0 = 3
  65. FluxOverflow = 4
  66. FluxUnderflow = 5
  67. Wrprot = 6
  68. NoUnit = 7
  69. NoBus = 8
  70. BadUnit = 9
  71. BadPin = 10
  72. str = {
  73. Okay: "Okay",
  74. BadCommand: "Bad Command",
  75. NoIndex: "No Index",
  76. NoTrk0: "Track 0 not found",
  77. FluxOverflow: "Flux Overflow",
  78. FluxUnderflow: "Flux Underflow",
  79. Wrprot: "Disk is Write Protected",
  80. NoUnit: "No drive unit selected",
  81. NoBus: "No bus type (eg. Shugart, IBM/PC) specified",
  82. BadUnit: "Bad unit number",
  83. BadPin: "Not a modifiable pin"
  84. }
  85. ## Cmd.GetInfo indexes
  86. class GetInfo:
  87. Firmware = 0
  88. BandwidthStats = 1
  89. ## Cmd.{Get,Set}Params indexes
  90. class Params:
  91. Delays = 0
  92. ## Cmd.SetBusType values
  93. class BusType:
  94. Invalid = 0
  95. IBMPC = 1
  96. Shugart = 2
  97. ## Flux read stream opcodes, preceded by 0xFF byte
  98. class FluxOp:
  99. LongFlux = 1
  100. Index = 2
  101. ## CmdError: Encapsulates a command acknowledgement.
  102. class CmdError(Exception):
  103. def __init__(self, cmd, code):
  104. self.cmd = cmd
  105. self.code = code
  106. def __str__(self):
  107. return "%s: %s" % (Cmd.str.get(self.cmd, "UnknownCmd"),
  108. Ack.str.get(self.code, "Unknown Error (%u)"
  109. % self.code))
  110. class Unit:
  111. ## Unit information, instance variables:
  112. ## major, minor: Greaseweazle firmware version number
  113. ## max_cmd: Maximum Cmd number accepted by this unit
  114. ## sample_freq: Resolution of all time values passed to/from this unit
  115. ## update_mode: True iff the Greaseweazle unit is in update mode
  116. ## Unit(ser):
  117. ## Accepts a Pyserial instance for Greaseweazle communications.
  118. def __init__(self, ser):
  119. self.ser = ser
  120. self.reset()
  121. # Copy firmware info to instance variables (see above for definitions).
  122. self._send_cmd(struct.pack("3B", Cmd.GetInfo, 3, GetInfo.Firmware))
  123. x = struct.unpack("<4BI3B21x", self.ser.read(32))
  124. (self.major, self.minor, is_main_firmware,
  125. self.max_cmd, self.sample_freq, self.hw_model,
  126. self.hw_submodel, self.usb_speed) = x
  127. # Old firmware doesn't report HW type but runs on STM32F1 only.
  128. if self.hw_model == 0:
  129. self.hw_model = 1
  130. # Check whether firmware is in update mode: limited command set if so.
  131. self.update_mode = (is_main_firmware == 0)
  132. if self.update_mode:
  133. self.update_jumpered = (self.sample_freq & 1)
  134. del self.sample_freq
  135. return
  136. # We are running main firmware: Check whether an update is needed.
  137. # We can use only the GetInfo command if the firmware is out of date.
  138. self.update_needed = (version.major != self.major
  139. or version.minor != self.minor)
  140. if self.update_needed:
  141. return
  142. # Initialise the delay properties with current firmware values.
  143. self._send_cmd(struct.pack("4B", Cmd.GetParams, 4, Params.Delays, 10))
  144. (self._select_delay, self._step_delay,
  145. self._seek_settle_delay, self._motor_delay,
  146. self._auto_off_delay) = struct.unpack("<5H", self.ser.read(10))
  147. ## reset:
  148. ## Resets communications with Greaseweazle.
  149. def reset(self):
  150. self.ser.reset_output_buffer()
  151. self.ser.baudrate = ControlCmd.ClearComms
  152. self.ser.baudrate = ControlCmd.Normal
  153. self.ser.reset_input_buffer()
  154. ## _send_cmd:
  155. ## Send given command byte sequence to Greaseweazle.
  156. ## Raise a CmdError if command fails.
  157. def _send_cmd(self, cmd):
  158. self.ser.write(cmd)
  159. (c,r) = struct.unpack("2B", self.ser.read(2))
  160. error.check(c == cmd[0], "Command returned garbage (%02x != %02x)"
  161. % (c, cmd[0]))
  162. if r != 0:
  163. raise CmdError(c, r)
  164. ## seek:
  165. ## Seek the selected drive's heads to the specified track (cyl, side).
  166. def seek(self, cyl, side):
  167. self._send_cmd(struct.pack("3B", Cmd.Seek, 3, cyl))
  168. self._send_cmd(struct.pack("3B", Cmd.Side, 3, side))
  169. ## set_bus_type:
  170. ## Set the floppy bus type.
  171. def set_bus_type(self, type):
  172. self._send_cmd(struct.pack("3B", Cmd.SetBusType, 3, type))
  173. ## set_pin:
  174. ## Set a pin level.
  175. def set_pin(self, pin, level):
  176. self._send_cmd(struct.pack("4B", Cmd.SetPin, 4, pin, int(level)))
  177. ## power_on_reset:
  178. ## Re-initialise to power-on defaults.
  179. def power_on_reset(self):
  180. self._send_cmd(struct.pack("2B", Cmd.Reset, 2))
  181. ## drive_select:
  182. ## Select the specified drive unit.
  183. def drive_select(self, unit):
  184. self._send_cmd(struct.pack("3B", Cmd.Select, 3, unit))
  185. ## drive_deselect:
  186. ## Deselect currently-selected drive unit (if any).
  187. def drive_deselect(self):
  188. self._send_cmd(struct.pack("2B", Cmd.Deselect, 2))
  189. ## drive_motor:
  190. ## Turn the specified drive's motor on/off.
  191. def drive_motor(self, unit, state):
  192. self._send_cmd(struct.pack("4B", Cmd.Motor, 4, unit, int(state)))
  193. ## switch_fw_mode:
  194. ## Switch between update bootloader and main firmware.
  195. def switch_fw_mode(self, mode):
  196. self._send_cmd(struct.pack("3B", Cmd.SwitchFwMode, 3, int(mode)))
  197. ## update_firmware:
  198. ## Update Greaseweazle to the given new firmware.
  199. def update_firmware(self, dat):
  200. self._send_cmd(struct.pack("<2BI", Cmd.Update, 6, len(dat)))
  201. self.ser.write(dat)
  202. (ack,) = struct.unpack("B", self.ser.read(1))
  203. return ack
  204. ## update_bootloader:
  205. ## Update Greaseweazle with the given new bootloader.
  206. def update_bootloader(self, dat):
  207. self._send_cmd(struct.pack("<2B2I", Cmd.Update, 10,
  208. len(dat), 0xdeafbee3))
  209. self.ser.write(dat)
  210. (ack,) = struct.unpack("B", self.ser.read(1))
  211. return ack
  212. ## _decode_flux:
  213. ## Decode the Greaseweazle data stream into a list of flux samples.
  214. def _decode_flux(self, dat):
  215. flux, index = [], []
  216. dat_i = iter(dat)
  217. ticks_since_index = 0
  218. try:
  219. while True:
  220. i = next(dat_i)
  221. if i < 250:
  222. flux.append(i)
  223. ticks_since_index += i
  224. elif i == 255:
  225. opcode = next(dat_i)
  226. if opcode == FluxOp.LongFlux:
  227. val = (next(dat_i) & 254) >> 1
  228. val += (next(dat_i) & 254) << 6
  229. val += (next(dat_i) & 254) << 13
  230. val += (next(dat_i) & 254) << 20
  231. flux.append(val)
  232. ticks_since_index += val
  233. elif opcode == FluxOp.Index:
  234. val = (next(dat_i) & 254) >> 1
  235. val += (next(dat_i) & 254) << 6
  236. val += (next(dat_i) & 254) << 13
  237. val += (next(dat_i) & 254) << 20
  238. index.append(ticks_since_index + val)
  239. ticks_since_index = -val
  240. pass
  241. else:
  242. raise error.Fatal("Bad opcode in flux stream (%d)"
  243. % opcode)
  244. else:
  245. val = 250 + (i - 250) * 255
  246. val += next(dat_i) - 1
  247. flux.append(val)
  248. ticks_since_index += val
  249. except StopIteration:
  250. pass
  251. error.check(flux[-1] == 0, "Missing terminator on flux read stream")
  252. return flux[:-1], index
  253. ## _encode_flux:
  254. ## Convert the given flux timings into an encoded data stream.
  255. def _encode_flux(self, flux):
  256. dat = bytearray()
  257. for val in flux:
  258. if val == 0:
  259. pass
  260. elif val < 250:
  261. dat.append(val)
  262. else:
  263. high = (val-250) // 255
  264. if high < 5:
  265. dat.append(250 + high)
  266. dat.append(1 + (val-250) % 255)
  267. else:
  268. dat.append(255)
  269. dat.append(FluxOp.LongFlux)
  270. dat.append(1 | (val<<1) & 255)
  271. dat.append(1 | (val>>6) & 255)
  272. dat.append(1 | (val>>13) & 255)
  273. dat.append(1 | (val>>20) & 255)
  274. dat.append(0) # End of Stream
  275. return dat
  276. ## _read_track:
  277. ## Private helper which issues command requests to Greaseweazle.
  278. def _read_track(self, nr_revs):
  279. # Request and read all flux timings for this track.
  280. dat = bytearray()
  281. self._send_cmd(struct.pack("<2BH", Cmd.ReadFlux, 4, nr_revs+1))
  282. while True:
  283. dat += self.ser.read(1)
  284. dat += self.ser.read(self.ser.in_waiting)
  285. if dat[-1] == 0:
  286. break
  287. # Check flux status. An exception is raised if there was an error.
  288. self._send_cmd(struct.pack("2B", Cmd.GetFluxStatus, 2))
  289. return dat
  290. ## read_track:
  291. ## Read and decode flux and index timings for the current track.
  292. def read_track(self, nr_revs, nr_retries=5):
  293. retry = 0
  294. while True:
  295. try:
  296. dat = self._read_track(nr_revs)
  297. except CmdError as error:
  298. # An error occurred. We may retry on transient overflows.
  299. if error.code == Ack.FluxOverflow and retry < nr_retries:
  300. retry += 1
  301. else:
  302. raise error
  303. else:
  304. # Success!
  305. break
  306. # Decode the flux list and read the index-times list.
  307. flux_list, index_list = self._decode_flux(dat)
  308. # Clip the initial partial revolution.
  309. to_index = index_list[0]
  310. for i in range(len(flux_list)):
  311. to_index -= flux_list[i]
  312. if to_index < 0:
  313. flux_list[i] = -to_index
  314. flux_list = flux_list[i:]
  315. break
  316. if to_index >= 0:
  317. # We ran out of flux.
  318. flux_list = []
  319. index_list = index_list[1:]
  320. # Success: Return the requested full index-to-index revolutions.
  321. return Flux(index_list, flux_list, self.sample_freq)
  322. ## write_track:
  323. ## Write the given flux stream to the current track via Greaseweazle.
  324. def write_track(self, flux_list, terminate_at_index, nr_retries=5):
  325. # Create encoded data stream.
  326. dat = self._encode_flux(flux_list)
  327. retry = 0
  328. while True:
  329. try:
  330. # Write the flux stream to the track via Greaseweazle.
  331. self._send_cmd(struct.pack("3B", Cmd.WriteFlux, 3,
  332. int(terminate_at_index)))
  333. self.ser.write(dat)
  334. self.ser.read(1) # Sync with Greaseweazle
  335. self._send_cmd(struct.pack("2B", Cmd.GetFluxStatus, 2))
  336. except CmdError as error:
  337. # An error occurred. We may retry on transient underflows.
  338. if error.code == Ack.FluxUnderflow and retry < nr_retries:
  339. retry += 1
  340. else:
  341. raise error
  342. else:
  343. # Success!
  344. break
  345. ## erase_track:
  346. ## Erase the current track via Greaseweazle.
  347. def erase_track(self, ticks):
  348. self._send_cmd(struct.pack("<2BI", Cmd.EraseFlux, 6, int(ticks)))
  349. self.ser.read(1) # Sync with Greaseweazle
  350. self._send_cmd(struct.pack("2B", Cmd.GetFluxStatus, 2))
  351. ## source_bytes:
  352. ## Command Greaseweazle to source 'nr' garbage bytes.
  353. def source_bytes(self, nr):
  354. self._send_cmd(struct.pack("<2BI", Cmd.SourceBytes, 6, nr))
  355. while nr > 0:
  356. self.ser.read(1)
  357. waiting = self.ser.in_waiting
  358. self.ser.read(waiting)
  359. nr -= 1 + waiting
  360. ## sink_bytes:
  361. ## Command Greaseweazle to sink 'nr' garbage bytes.
  362. def sink_bytes(self, nr):
  363. self._send_cmd(struct.pack("<2BI", Cmd.SinkBytes, 6, nr))
  364. dat = bytes(1024*1024)
  365. while nr > len(dat):
  366. self.ser.write(dat)
  367. nr -= len(dat)
  368. self.ser.write(dat[:nr])
  369. self.ser.read(1) # Sync with Greaseweazle
  370. ## bw_stats:
  371. ## Get min/max bandwidth for previous source/sink command. Mbps (float).
  372. def bw_stats(self):
  373. self._send_cmd(struct.pack("3B", Cmd.GetInfo, 3,
  374. GetInfo.BandwidthStats))
  375. min_bytes, min_usecs, max_bytes, max_usecs = struct.unpack(
  376. "<4I16x", self.ser.read(32))
  377. min_bw = (8 * min_bytes) / min_usecs
  378. max_bw = (8 * max_bytes) / max_usecs
  379. return min_bw, max_bw
  380. ##
  381. ## Delay-property public getters and setters:
  382. ## select_delay: Delay (usec) after asserting drive select
  383. ## step_delay: Delay (usec) after issuing a head-step command
  384. ## seek_settle_delay: Delay (msec) after completing a head-seek operation
  385. ## motor_delay: Delay (msec) after turning on drive spindle motor
  386. ## auto_off_delay: Timeout (msec) since last command upon which all
  387. ## drives are deselected and spindle motors turned off
  388. ##
  389. def _set_delays(self):
  390. self._send_cmd(struct.pack("<3B5H", Cmd.SetParams,
  391. 3+5*2, Params.Delays,
  392. self._select_delay, self._step_delay,
  393. self._seek_settle_delay,
  394. self._motor_delay, self._auto_off_delay))
  395. @property
  396. def select_delay(self):
  397. return self._select_delay
  398. @select_delay.setter
  399. def select_delay(self, select_delay):
  400. self._select_delay = select_delay
  401. self._set_delays()
  402. @property
  403. def step_delay(self):
  404. return self._step_delay
  405. @step_delay.setter
  406. def step_delay(self, step_delay):
  407. self._step_delay = step_delay
  408. self._set_delays()
  409. @property
  410. def seek_settle_delay(self):
  411. return self._seek_settle_delay
  412. @seek_settle_delay.setter
  413. def seek_settle_delay(self, seek_settle_delay):
  414. self._seek_settle_delay = seek_settle_delay
  415. self._set_delays()
  416. @property
  417. def motor_delay(self):
  418. return self._motor_delay
  419. @motor_delay.setter
  420. def motor_delay(self, motor_delay):
  421. self._motor_delay = motor_delay
  422. self._set_delays()
  423. @property
  424. def auto_off_delay(self):
  425. return self._auto_off_delay
  426. @auto_off_delay.setter
  427. def auto_off_delay(self, auto_off_delay):
  428. self._auto_off_delay = auto_off_delay
  429. self._set_delays()
  430. # Local variables:
  431. # python-indent: 4
  432. # End: