gd407使用dm9000通讯异常

如题,gd407使用dm9000,当通讯一段时间后网络掉线,无法连接,程序未跑死,必须重启或者重新配置dm9000才能重新连接。经过排查之后,在dm9000的发送函数添加一小段延时解决

rt_err_t rt_dm9000_tx( rt_device_t dev, struct pbuf *p)
{
	uint64 tx_timestamp	=	0;
  int tx_status = 0;
  DM9000_TRACE("dm9000 tx: %d\n", p->tot_len);

  /* lock DM9000 device */
  rt_sem_take(&sem_lock, RT_WAITING_FOREVER);

  /* disable dm9000a interrupt */
  dm9000_io_write(DM9000_IMR, IMR_PAR);

  /* Move data to DM9000 TX RAM */
  DM9000_outb(DM9000_IO_BASE, DM9000_MWCMD);

  {
      /* q traverses through linked list of pbuf's
       * This list MUST consist of a single packet ONLY */
      struct pbuf *q;
      rt_uint16_t pbuf_index = 0;
      rt_uint8_t word[2], word_index = 0;

      q = p;
      /* Write data into dm9000a, two bytes at a time
       * Handling pbuf's with odd number of bytes correctly
       * No attempt to optimize for speed has been made */
      while (q)
      {
          if (pbuf_index < q->len)
          {
              word[word_index++] = ((u8_t *)q->payload)[pbuf_index++];
              if (word_index == 2)
              {
                  DM9000_outw(DM9000_DATA_BASE, (word[1] << 8) | word[0]);
                  word_index = 0;
                  dm9000_delay(5); //添加小段延时,约2us
              }
          }
          else
          {
              q = q->next;
              pbuf_index = 0;
          }
      }
      /* One byte could still be unsent */
      if (word_index == 1)
      {
          DM9000_outw(DM9000_DATA_BASE, word[0]);
      }
  }

  if (dm9000_device.packet_cnt == 0)
  {
      DM9000_TRACE("dm9000 tx: first packet\n");

      dm9000_device.packet_cnt ++;
      /* Set TX length to DM9000 */
      dm9000_io_write(DM9000_TXPLL, p->tot_len & 0xff);
      dm9000_io_write(DM9000_TXPLH, (p->tot_len >> 8) & 0xff);

      /* Issue TX polling command */
      dm9000_io_write(DM9000_TCR, TCR_TXREQ);	/* Cleared after TX complete */
  }
  else
  {
      DM9000_TRACE("dm9000 tx: second packet\n");

      dm9000_device.packet_cnt ++;
      dm9000_device.queue_packet_len = p->tot_len;
  }

  /* enable dm9000a interrupt */
  dm9000_io_write(DM9000_IMR, dm9000_device.imr_all);

  /* unlock DM9000 device */
  rt_sem_release(&sem_lock);


	tx_timestamp	=	sys_timer_Getstamp();
  
   /* transmit done */
  while(1)
  {
    //rt_thread_delay(1);
    dm9000_delay(5); //添加小段延时,约2us
    tx_status = dm9000_io_read(DM9000_NSR);    /* Got TX status */

    if (tx_status & (NSR_TX2END | NSR_TX1END))
    {
        txcount++;
        dm9000_device.packet_cnt --;
        if (dm9000_device.packet_cnt > 0)
        {
            DM9000_TRACE("dm9000 isr: tx second packet\n");

            /* transmit packet II */
            /* Set TX length to DM9000 */
            dm9000_io_write(DM9000_TXPLL, dm9000_device.queue_packet_len & 0xff);
            dm9000_io_write(DM9000_TXPLH, (dm9000_device.queue_packet_len >> 8) & 0xff);

            /* Issue TX polling command */
            dm9000_io_write(DM9000_TCR, TCR_TXREQ);	/* Cleared after TX complete */
        }
        break;
    }
    else 
    {
      txrecount++;
      if(sys_timer_getstamp_between(tx_timestamp) >=	20000)
      {
        dm9000_device.packet_txtimecout++;
        dm9000_device.packet_cnt --;
        if (dm9000_device.packet_cnt > 0)
        {
            DM9000_TRACE("dm9000 isr: tx second packet\n");
            
            /* transmit packet II */
            /* Set TX length to DM9000 */
            dm9000_io_write(DM9000_TXPLL, dm9000_device.queue_packet_len & 0xff);
            dm9000_io_write(DM9000_TXPLH, (dm9000_device.queue_packet_len >> 8) & 0xff);

            /* Issue TX polling command */
            dm9000_io_write(DM9000_TCR, TCR_TXREQ);	/* Cleared after TX complete */
        }
        break;
      }
    }
  }
  DM9000_TRACE("dm9000 tx done\n");

    return RT_EOK;
}

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

大文梅

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值