ymodem.h 4.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135
  1. #ifndef __YMODEM_H__
  2. #define __YMODEM_H__
  3. /*
  4. * File : ymodem.h
  5. * COPYRIGHT (C) 2012, Shanghai Real-Thread Technology Co., Ltd
  6. *
  7. * Change Logs:
  8. * Date Author Notes
  9. * 2013-04-14 Grissiom initial implementation
  10. */
  11. #include "rtthread.h"
  12. /* The word "RYM" is stand for "Real-YModem". */
  13. enum rym_code {
  14. RYM_CODE_NONE = 0x00,
  15. RYM_CODE_SOH = 0x01,
  16. RYM_CODE_STX = 0x02,
  17. RYM_CODE_EOT = 0x04,
  18. RYM_CODE_ACK = 0x06,
  19. RYM_CODE_NAK = 0x15,
  20. RYM_CODE_CAN = 0x18,
  21. RYM_CODE_C = 0x43,
  22. };
  23. /* RYM error code
  24. *
  25. * We use the rt_err_t to return error values. We take use of current error
  26. * codes available in RTT and append ourselves.
  27. */
  28. /* timeout on handshake */
  29. #define RYM_ERR_TMO 0x70
  30. /* wrong code, wrong SOH, STX etc. */
  31. #define RYM_ERR_CODE 0x71
  32. /* wrong sequence number */
  33. #define RYM_ERR_SEQ 0x72
  34. /* wrong CRC checksum */
  35. #define RYM_ERR_CRC 0x73
  36. /* not enough data received */
  37. #define RYM_ERR_DSZ 0x74
  38. /* the transmission is aborted by user */
  39. #define RYM_ERR_CAN 0x75
  40. /* how many ticks wait for chars between packet. */
  41. #ifndef RYM_WAIT_CHR_TICK
  42. #define RYM_WAIT_CHR_TICK (RT_TICK_PER_SECOND * 3)
  43. #endif
  44. /* how many ticks wait for between packet. */
  45. #ifndef RYM_WAIT_PKG_TICK
  46. #define RYM_WAIT_PKG_TICK (RT_TICK_PER_SECOND * 3)
  47. #endif
  48. /* how many ticks between two handshake code. */
  49. #ifndef RYM_CHD_INTV_TICK
  50. #define RYM_CHD_INTV_TICK (RT_TICK_PER_SECOND * 3)
  51. #endif
  52. enum rym_stage {
  53. RYM_STAGE_NONE,
  54. /* set when C is send */
  55. RYM_STAGE_ESTABLISHING,
  56. /* set when we've got the packet 0 and sent ACK and second C */
  57. RYM_STAGE_ESTABLISHED,
  58. /* set when the sender respond to our second C */
  59. RYM_STAGE_TRANSMITTING,
  60. /* set when the sender send a EOT */
  61. RYM_STAGE_FINISHING,
  62. /* set when transmission is really finished, i.e., after the NAK, C, final
  63. * NULL packet stuff. */
  64. RYM_STAGE_FINISHED,
  65. };
  66. struct rym_ctx;
  67. /* when receiving files, the buf will be the data received from ymodem protocol
  68. * and the len is the data size.
  69. *
  70. * TODO:
  71. * When sending files, the len is the buf size in RYM. The callback need to
  72. * fill the buf with data to send. Returning RYM_CODE_EOT will terminate the
  73. * transfer and the buf will be discarded. Any other return values will cause
  74. * the transfer continue.
  75. */
  76. typedef enum rym_code (*rym_callback)(struct rym_ctx *ctx, rt_uint8_t *buf, rt_size_t len);
  77. /* Currently RYM only support one transfer session(ctx) for simplicity.
  78. *
  79. * In case we could support multiple sessions in The future, the first argument
  80. * of APIs are (struct rym_ctx*).
  81. */
  82. struct rym_ctx
  83. {
  84. rym_callback on_data;
  85. rym_callback on_begin;
  86. rym_callback on_end;
  87. /* When error happened, user need to check this to get when the error has
  88. * happened. */
  89. enum rym_stage stage;
  90. /* user could get the error content through this */
  91. rt_uint8_t *buf;
  92. struct rt_semaphore sem;
  93. rt_device_t dev;
  94. };
  95. /** recv a file on device dev with ymodem session ctx.
  96. *
  97. * If an error happens, you can get where it is failed from ctx->stage.
  98. *
  99. * @param on_begin The callback will be invoked when the first packet arrived.
  100. * This packet often contain file names and the size of the file, if the sender
  101. * support it. So if you want to save the data to a file, you may need to
  102. * create the file on need. It is the on_begin's responsibility to parse the
  103. * data content. The on_begin can be NULL, in which case the transmission will
  104. * continue without any side-effects.
  105. *
  106. * @param on_data The callback will be invoked on the packets received. The
  107. * callback should save the data to the destination. The return value will be
  108. * sent to the sender and in turn, only RYM_{ACK,CAN} is valid. When on_data is
  109. * NULL, RYM will barely send ACK on every packet and have no side-effects.
  110. *
  111. * @param on_end The callback will be invoked when one transmission is
  112. * finished. The data should be 128 bytes of NULL. You can do some cleaning job
  113. * in this callback such as closing the file. The return value of this callback
  114. * is ignored. As above, this parameter can be NULL if you don't need such
  115. * function.
  116. *
  117. * @param handshake_timeout the timeout when hand shaking. The unit is in
  118. * second.
  119. */
  120. rt_err_t rym_recv_on_device(struct rym_ctx *ctx, rt_device_t dev,
  121. rym_callback on_begin, rym_callback on_data, rym_callback on_end,
  122. int handshake_timeout);
  123. #endif