⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 stream.c

📁 <B>SMSC USB2.0 Flash硬盘驱动源码</B>
💻 C
字号:
/*============================================================================
  ____________________________________________________________________________
                                ______________________________________________
   SSSS  M   M          CCCC          Standard Microsystems Corporation
  S      MM MM   SSSS  C                    Austin Design Center
   SSS   M M M  S      C                 11000 N. Mopac Expressway
      S  M   M   SSS   C                Stonelake Bldg. 6, Suite 500
  SSSS   M   M      S   CCCC                Austin, Texas 78759
                SSSS            ______________________________________________
  ____________________________________________________________________________
  Copyright(C) 1999, Standard Microsystems Corporation
  All Rights Reserved.
  This program code listing is proprietary to SMSC and may not be copied,
  distributed, or used without a license to do so.  Such license may have
  Limited or Restricted Rights. Please refer to the license for further
  clarification.
  ____________________________________________________________________________
  Notice: The program contained in this listing is a proprietary trade
  secret of SMSC, Hauppauge, New York, and is copyrighted
  under the United States Copyright Act of 1976 as an unpublished work,
  pursuant to Section 104 and Section 408 of Title XVII of the United
  States code. Unauthorized copying, adaption, distribution, use, or
  display is prohibited by this law.
  ____________________________________________________________________________
  Use, duplication, or disclosure by the Government is subject to
  restrictions as set forth in subparagraph(c)(1)(ii) of the Rights
  in Technical Data and Computer Software clause at DFARS 52.227-7013.
  Contractor/Manufacturer is Standard Microsystems Corporation,
  80 Arkay Drive, Hauppauge, New York, 1178-8847.
  ____________________________________________________________________________
  ____________________________________________________________________________
  stream.c - pseudo-stream implmentation over usb endpoint
  ____________________________________________________________________________
  comments tbd
  ____________________________________________________________________________
  Revision History
  Date      Who  Comment
  ________  ___  _____________________________________________________________
  06/09/01  cds  initial version
============================================================================*/
#include "project.h"

//------------------------------------------------------------------------------
// stream state machine globals
static xdata   uint16  _stream_last_pnr ;
static xdata   uint8   _stream_2_dir ;
static xdata   uint8   _stream_2_pnr ;
static xdata   uint16  _stream_2_offset ;
static xdata   uint16  _stream_2_len ;


//------------------------------------------------------------------------------
// Declaration:
//  void stream_open(uint8 ndp, uint8 dir) reentrant;
// Purpose:
//  opens the given endpoint for streaming bytes in the given direction.
// Arguments:
//  ndp - the endpoint to be used for streaming the bytes
//  dir - the direction to stream.  k_data_in (dev to host) or k_data_out (host to dev)
// Return:
//  null
//------------------------------------------------------------------------------
void stream_open(uint8 ndp, uint8 dir) reentrant
{
  uint8 tmp = ndp;
  _stream_2_dir    = dir ;
  _stream_2_offset = 0 ;
  _stream_2_len    = 0 ;
  if (_stream_2_dir == k_data_out)
  {
    trace0(0, stream, 0, "ndp:2 open for data-out") ;
    // enable the buffer now
    _buffer_rx_enable(4);
  }
  else
  {
    trace0(0, stream, 0, "ndp:2 open for data-in") ;
    _stream_2_pnr    = 4 ;
    _stream_2_offset = 0 ;
    _stream_2_len    = ((x_sie_conf & kbm_sie_conf_hspeed)) ? k_hs_maxpktsz : k_fs_maxpktsz ;
    x_ramrdbc_a2      = _l(0);
    x_ramrdbc_a1      = _h(0);
  }
}

//------------------------------------------------------------------------------
// Declaration:
//  uint8 stream_get(uint8* bytep) reentrant;
// Purpose:
//  Gets the next byte from an k_data_out stream.
// Arguments:
//  ndp - the stream ndp
//  bytep - a pointer to where the byte will be placed
// Return:
//  0 - no bytes available.
//  1 - one byte placed into bytep location
//------------------------------------------------------------------------------
uint8 stream_put(uint8 ndp, uint8 byte) reentrant
{
  uint16 pkt_offset ;
  ndp=ndp ;
  if (_stream_2_dir != k_data_in)
  {
    // stream opened for data-out.  cannot put byte into stream
    return 0;
  }
  if (!(_stream_2_len))
  {
    // no current packet - check if the next one is available...
    uint8 tmppnr = ((_stream_2_pnr==4)?5:4) ;
    trace1(0, stream, 0, "ndp 2: try to allocate buffer %c", (tmppnr-4+'a'));
    if (!(packet_is_available(tmppnr)))
    {
      // packet not available.  no room for byte at this time
      return 0 ;
    }
    // buffer is available
    _stream_2_pnr    = tmppnr ;
    _stream_2_offset = 0 ;
    _stream_2_len    = ((x_sie_conf & kbm_sie_conf_hspeed)) ? k_hs_maxpktsz : k_fs_maxpktsz ;
  }
  // we've got room for at least one byte.  put it into the packet buffer
  mcu_begin_critical_section() ;
  // the last put to sram was to a different pnr (buffer) location.  reset the offset
  pkt_offset = _uint16(k_pkt_addrhi[_stream_2_pnr], k_pkt_addrlo[_stream_2_pnr]) + _stream_2_offset ;
  x_sram_addr_lo = _l(pkt_offset) ;
  x_sram_addr_hi = _h(pkt_offset) ;
  // sram addr is now correct
  x_sram_data = byte ;
  // update bytecount
  _stream_2_offset++ ;
  mcu_end_critical_section();
  // check if the packet buffer is now full
  if (_stream_2_offset == _stream_2_len)
  {
    // transmit it
    if (_stream_2_pnr == 4)  // buffer A
    {
      trace1(0, stream, 0, "ndp:2, buf:a - transmitting %d bytes",_stream_2_len);
      x_ramrdbc_a2 = _l(_stream_2_len);
      x_ramrdbc_a1 = _h(_stream_2_len);
    }
    // not using an else right now to help debug
    if (_stream_2_pnr == 5)  // buffer B
    {
      trace1(0, stream, 0, "ndp:2, buf:b - transmitting %d bytes",_stream_2_len);
      x_ramrdbc_b2 = _l(_stream_2_len);
      x_ramrdbc_b1 = _h(_stream_2_len);
    }
    // clear offset and length so that next pass, we try to grab the next buffer
    _stream_2_offset = _stream_2_len = 0 ;
    txfifo_wr( 2, _stream_2_pnr ) ;
  }
  // report our byte
  return 1;
}

//------------------------------------------------------------------------------
// Declaration:
//  uint8 stream_peek(uint8 ndp, uint8* bytep) reentrant ;
// Purpose:
//  Peeks at the stream and returns a byte if one is available,
//  but does not remove the byte from the stream.  This function
//  can be used to ensure that there are bytes available to be
//  "gotten" before moving on.
// Arguments:
//  ndp - the stream ndp
//  bytep - ptr to where byte will be stored
// Return:
//  0 - no bytes available.
//  1 - one byte placed into bytep location
//------------------------------------------------------------------------------
uint8 stream_peek(uint8 ndp, uint8* bytep) reentrant
{
  uint16 pkt_offset ;
  ndp=ndp ;
  if (_stream_2_dir != k_data_out)
  {
    return 0 ;
  }
  if (!_stream_2_len)
  {
    // no current packet - check to see if one has come in
    uint8 tmppnr = endpoint_rd_rx_pnr(k_rx_pipe);
    uint8 ch ;
    if (!tmppnr)
    {
      // no packets have arrived
      trace0(0, stream, 0, "ndp:2 get failed - no packets") ;
      return 0 ;
    }
    // buffer is available
    _stream_2_pnr    = tmppnr ;
    _stream_2_offset = 0 ;
    _stream_2_len    = mmu_rd_len(_stream_2_pnr) ;
    ch=(_stream_2_pnr-4+'a') ;
    trace2(0, stream, 0, "ndp:2 buf:%c - received %d bytes", ch, _stream_2_len) ;
  }
  // we've got room for at least one byte.  put it into the packet buffer
  mcu_begin_critical_section() ;
  // the last put to sram was to a different pnr (buffer) location.  reset the offset
  pkt_offset = _uint16(k_pkt_addrhi[_stream_2_pnr], k_pkt_addrlo[_stream_2_pnr]) + _stream_2_offset ;
  x_sram_addr_lo = _l(pkt_offset) ;
  x_sram_addr_hi = _h(pkt_offset) ;
  // read the data
  *bytep=x_sram_data ;
  mcu_end_critical_section() ;
  return 1 ;
}

//------------------------------------------------------------------------------
// Declaration:
//  void stream_put(uint8 ndp, uint8 byte);
// Purpose:
//  Put one byte on the out-bound (k_data_in) stream and send
//  off the packet if the packet is full.
// Arguments:
//  ndp - the stream ndp
//  byte - data to put
// Return:
//  0 - no space for byte.  try again later.
//  1 - one byte put into the stream.
//------------------------------------------------------------------------------
uint8 stream_get(uint8 ndp, uint8* bytep) reentrant
{
  uint16 pkt_offset ;
  ndp=ndp ;
  if (_stream_2_dir != k_data_out)
  {
    return 0 ;
  }
  if (!_stream_2_len)
  {
    // no current packet - check to see if one has come in
    uint8 tmppnr = endpoint_rd_rx_pnr(k_rx_pipe);
    uint8 ch ;
    if (!tmppnr)
    {
      // no packets have arrived
      trace0(0, stream, 0, "ndp:2 get failed - no packets") ;
      return 0 ;
    }
    // buffer is available
    _stream_2_pnr    = tmppnr ;
    _stream_2_offset = 0 ;
    _stream_2_len    = mmu_rd_len(_stream_2_pnr) ;
    ch=(_stream_2_pnr-4+'a') ;
    trace2(0, stream, 0, "ndp:2 buf:%c - received %d bytes", ch, _stream_2_len) ;
  }
  // we've got room for at least one byte.  put it into the packet buffer
  mcu_begin_critical_section() ;
  // the last put to sram was to a different pnr (buffer) location.  reset the offset
  pkt_offset = _uint16(k_pkt_addrhi[_stream_2_pnr], k_pkt_addrlo[_stream_2_pnr]) + _stream_2_offset ;
  x_sram_addr_lo = _l(pkt_offset) ;
  x_sram_addr_hi = _h(pkt_offset) ;
  // read the data
  *bytep=x_sram_data ;
  _stream_2_offset++ ;
  mcu_end_critical_section() ;
  // free packet if complete
  if (_stream_2_offset == _stream_2_len)
  {
    mmu_deallocate(_stream_2_pnr);
    _buffer_rx_enable(_stream_2_pnr);
    _stream_2_len=0 ;
  }
  return 1 ;
}

//------------------------------------------------------------------------------
// Declaration:
//  void stream_flush(uint8 ndp) reentrant
// Purpose:
//  flushes whatever data is in the stream out to the sie.  if
//  there are 0 bytes in the current pkt, a ZLP is sent over
//  the usb.
// Arguments:
//  ndp - the stream ndp to flush
// Return:
//  none
//------------------------------------------------------------------------------
void stream_flush(uint8 ndp) reentrant
{
  ndp = ndp ;
  trace1(0, stream, 1, "stream_flush(%d)", ndp) ;
  if (_stream_2_dir == k_data_in)
  {
    mcu_begin_critical_section() ;
    // transmit anything (incl zlp) in the buffer
    if (_stream_2_pnr == 4)  // buffer A
    {
      trace1(0, stream, 0, "ndp:2, buf:a - flushing %d bytes",_stream_2_offset);
      x_ramrdbc_a2 = _l(_stream_2_offset);
      x_ramrdbc_a1 = _h(_stream_2_offset);
    }
    // not using an else right now to help debug
    if (_stream_2_pnr == 5)  // buffer B
    {
      trace1(0, stream, 0, "ndp:2, buf:b - flushing %d bytes",_stream_2_offset);
      x_ramrdbc_b2 = _l(_stream_2_offset);
      x_ramrdbc_b1 = _h(_stream_2_offset);
    }
    // clear offset and length so that next pass, we try to grab the next buffer
    _stream_2_offset = _stream_2_len = 0 ;
    mcu_end_critical_section();
    txfifo_wr( 2, _stream_2_pnr ) ;
  }
  else
  {
    // data out - simply free the pnr
    // free packet if complete
    if (_stream_2_len)
    {
      trace1(0, stream, 0, "ndp:2 pnr:%d - flushing %d bytes", _stream_2_pnr) ;
      mmu_deallocate(_stream_2_len);
      _stream_2_len=0 ;
    }
  }
}

//------------------------------------------------------------------------------
// Declaration:
//  void stream_close(uint8 ndp) reentrant ;
// Purpose:
//  closes an ndp for streaming io.  if there is data remaining
//  on a data_in stream, it is tx'd before this fn returns.
//  unlike 'flush', if there is 0 data, a zlp is _not_ sent.
// Arguments:
//  ndp - endpoint to close
// Return:
//  none
//------------------------------------------------------------------------------
void stream_close(uint8 ndp) reentrant
{
  trace1(0, stream, 0, "close(%d)", ndp) ;
  // if there is any data... send it out... if there is NOT, don't send a zlp
  // if this is a data-out, always flush (to free the packets)
  if (_stream_2_offset || (_stream_2_dir==k_data_out)) stream_flush(ndp) ;
  // clear the stream info just for s&g
  _stream_2_dir    = 0 ;
  _stream_2_offset = 0 ;
  _stream_2_len    = 0 ;
  _stream_2_pnr    = 0 ;
}

//---eof------------------------------------------------------------------------

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -