OpenCores
URL https://opencores.org/ocsvn/openrisc/openrisc/trunk

Subversion Repositories openrisc

[/] [openrisc/] [trunk/] [rtos/] [ecos-3.0/] [packages/] [io/] [usb/] [serial/] [slave/] [current/] [tests/] [usb2serial.c] - Rev 867

Go to most recent revision | Compare with Previous | Blame | View Log

//==========================================================================
//
//      usb2serial.c
//
//      Example application for the USB serial layer in eCos.
//
//==========================================================================
// ####ECOSGPLCOPYRIGHTBEGIN####                                            
// -------------------------------------------                              
// This file is part of eCos, the Embedded Configurable Operating System.   
// Copyright (C) 2008, 2010 Free Software Foundation, Inc.                        
//
// eCos is free software; you can redistribute it and/or modify it under    
// the terms of the GNU General Public License as published by the Free     
// Software Foundation; either version 2 or (at your option) any later      
// version.                                                                 
//
// eCos is distributed in the hope that it will be useful, but WITHOUT      
// ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or    
// FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License    
// for more details.                                                        
//
// You should have received a copy of the GNU General Public License        
// along with eCos; if not, write to the Free Software Foundation, Inc.,    
// 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301, USA.            
//
// As a special exception, if other files instantiate templates or use      
// macros or inline functions from this file, or you compile this file      
// and link it with other works to produce a work based on this file,       
// this file does not by itself cause the resulting work to be covered by   
// the GNU General Public License. However the source code for this file    
// must still be made available in accordance with section (3) of the GNU   
// General Public License v2.                                               
//
// This exception does not invalidate any other reasons why a work based    
// on this file might be covered by the GNU General Public License.         
// -------------------------------------------                              
// ####ECOSGPLCOPYRIGHTEND####                                              
//===========================================================================
//#####DESCRIPTIONBEGIN####
//
// Author(s):    Frank M. Pagliughi (fmp), SoRo Systems, Inc.
// Contributors: 
// Date:         2008-06-02
// Description:  USB serial example application.
//
//####DESCRIPTIONEND####
//===========================================================================
 
#include <cyg/kernel/kapi.h>
#include <cyg/hal/hal_arch.h>
#include <cyg/infra/diag.h>
#include <pkgconf/kernel.h>
#include <cyg/io/serialio.h>
 
#include <cyg/io/usb/usbs_serial.h>
#include <pkgconf/io_usb_slave_serial.h>
 
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <fcntl.h>
#include <sys/stat.h>
 
// This application creates a USB-serial converter. To the host it will appear
// as a single serial USB port, such as /dev/ttyUSB0 for a Linux host.
// Any characters received from the USB host will be sent out the serial port
// and visa-verse. It creates a separate, dedicated thread for each direction.
// 
// It uses the eCos USB-serial layer to enumerate with the USB host and monitor
// the connection, but then uses standard C I/O functions to perform the 
// communications.
// 
// The USB serial module can be configured as a generic adapter or an an ACM
// communications class device. For the latter, the application handles the
// USB communications class requests which allows it to receive requests from
// the host to set serial parameters, like the baud rate. This actually turns
// this example into a more realistic USB-serial adapter that can be configured
// dynamically by the host.
// 
// The eCos library must be configured with the packages for USB slave, USB 
// serial, and File I/O. It also requires the proper serial port driver for the
// target platform.
// 
// This example was tested with the AT91SAM7S-EK board, but should work with any
// board that has a USB slave and serial port, and the necessary drivers.
 
 
// Comment this line out to remove debug output.
#define DEBUG_OUTPUT
 
#if defined(DEBUG_OUTPUT)
#define DBG diag_printf
#else
#define DBG (1) ? (void)0 : diag_printf
#endif
 
// Set these to the USB devtab entries for the Tx and Rx Bulk endpoints 
// selected in the configuration of the USB serial subsystem.
#define USB_TX_DEV  "/dev/usbs1"
#define USB_RX_DEV  "/dev/usbs2"
 
// Set this for any available serial port on the target.
#define SER_DEV     "/dev/ser0"
 
// Buffer for incoming USB bulk data. The local USB driver can probably split
// packets, but just in case, making this the page size of the host might be
// helpful.
#define BUF_SIZE 4096
static char usb2ser_buf[BUF_SIZE];
 
// The threads
cyg_thread thread[2];
 
// Space for two 4K stacks
#define THREAD_STACK_SIZE 4096
char stack[2][THREAD_STACK_SIZE];
 
// The handles for the threads
cyg_handle_t    usb2ser_thread, 
                ser2usb_thread;
 
// --------------------------------------------------------------------------
// For an ACM serial device we can handle the USB class messages to deal with
// requests from the host like setting the serial parameters (baud rate, 
// etc).
 
#ifdef CYGDAT_IO_USB_SLAVE_CLASS_TYPE_ACM
 
static cyg_uint8    acm_buf[32];
static cyg_uint32   baud = 34800;
 
// --------------------------------------------------------------------------
// Handler for the completion of a SetLineCoding request from the host.
// The 'acm_buf' should contain the 7-byte request OUT packet from the
// host. This contains a request to change the serial parameters.
// In this example function, we will accept a few different baud rates.
// To keep the example relatively simple, though, we keep the other serial
// parameters to 1 stop bit, no parity, 8 data bits.
 
static usbs_control_return
acm_set_line_coding(usbs_control_endpoint* ep0, int n)
{
  int err;
  cyg_uint32 req_baud;
  cyg_io_handle_t handle;
 
  // Get the requested baud rate from the received ctrl OUT packet
  req_baud = ((acm_buf[3] << 24) | (acm_buf[2] << 16) | 
              (acm_buf[1] << 8) | acm_buf[0]);
 
  DBG("Set Baud: %u\n", (unsigned) baud);
 
  // Look up the serial handle and attempt to set the baud rate.
  if (cyg_io_lookup(SER_DEV, &handle) == 0) {
    cyg_serial_info_t ser_info;
    cyg_uint32 len = sizeof(ser_info);
 
    switch (baud) {
      case   9600 : ser_info.baud = CYGNUM_SERIAL_BAUD_9600;      break;
      case  38400 : ser_info.baud = CYGNUM_SERIAL_BAUD_38400;     break;
      case 115200 : ser_info.baud = CYGNUM_SERIAL_BAUD_115200;    break;
      default:
        DBG("Unsupported baud rate\n");
        return USBS_CONTROL_RETURN_HANDLED;
    }
    ser_info.stop = CYGNUM_SERIAL_STOP_1;
    ser_info.parity = CYGNUM_SERIAL_PARITY_NONE;
    ser_info.word_length = CYGNUM_SERIAL_WORD_LENGTH_8;
    ser_info.flags = 0;
 
    err = cyg_io_set_config(handle, CYG_IO_SET_CONFIG_SERIAL_INFO, 
                            &ser_info, &len);
    if (err == 0)
      baud = req_baud;
    else {
      DBG("Error setting serial params\n");
    }
  }
  else {
    DBG("Error looking up serial device: %s\n", SER_DEV);
  }
  return USBS_CONTROL_RETURN_HANDLED;
}
 
// --------------------------------------------------------------------------
// Handler for the ACM class messages.
// 
static usbs_control_return 
acm_class_handler(usbs_control_endpoint* ep0, void* data)
{
  usbs_control_return result = USBS_CONTROL_RETURN_UNKNOWN;
 
  usb_devreq  *req = (usb_devreq *) ep0->control_buffer;
 
  static cyg_uint8 rsp_buf[32];
 
  DBG("ACM Class Handler\n");
 
  switch (req->request) {
 
    case USBS_SERIAL_SET_LINE_CODING :
      DBG("Set Line Coding\n");
      memset(acm_buf, 0, 32);
      ep0->buffer = acm_buf;
      ep0->buffer_size = 7;
      ep0->complete_fn = acm_set_line_coding;
      result = USBS_CONTROL_RETURN_HANDLED;
      break;
 
    case USBS_SERIAL_GET_LINE_CODING :
      DBG("Get Line Coding\n");
      rsp_buf[0] = baud & 0xFF;
      rsp_buf[1] = (baud >>  8) & 0xFF;
      rsp_buf[2] = (baud >> 16) & 0xFF;
      rsp_buf[3] = (baud >> 24) & 0xFF;
      rsp_buf[4] = 0; // One stop bit
      rsp_buf[5] = 0; // No parity
      rsp_buf[6] = 8; // 8 data bits
      ep0->buffer = rsp_buf;
      ep0->buffer_size = 7;
      result = USBS_CONTROL_RETURN_HANDLED;
      break;
 
    default :
      DBG("*** Unhandled ACM Request: 0x%02X ***\n",
          (unsigned) req->request);
  }
 
  return result;
}
 
#endif      // CYGDAT_IO_USB_SLAVE_CLASS_TYPE_ACM
 
// --------------------------------------------------------------------------
// Thread receives packets from the USB and sends them out the serial port
// It uses a buffered stdio input, an un-buffered low-level file output.
// This isn't terribly efficient, but rather an example of both methods.
 
void usb2ser_func(cyg_addrword_t data)
{
  int  c;
  FILE *rxf = fopen(USB_RX_DEV, "r");
  int   txh = open(SER_DEV, O_WRONLY, 0);
 
  DBG("Usb2Ser: Thread starting\n");
 
  if (!rxf) {
    DBG("Error opening USB rx port\n");
    return;
  }
 
  if (txh < 0) {
    DBG("Error opening serial tx port\n");
    return;
  }
 
  // Give the USB receiver an adequate buffer.
  setvbuf(rxf, usb2ser_buf, _IOFBF, BUF_SIZE);
 
  while (1) {
 
    // ----- Wait for the host to configure -----
 
    DBG("Usb2Ser: Waiting for USB configuration\n");
    usbs_serial_wait_until_configured();
    cyg_thread_delay((cyg_tick_count_t) 10);
 
    // ----- While configured read data & send out serial port -----
 
    DBG("Usb2Ser: USB configured\n");
    while (usbs_serial_is_configured()) {
      if ((c = getc(rxf)) < 0) {
        DBG("*** USB Read Error: %d ***\n", c);
      }
      else {
        char ch = (char) c;
        write(txh, &ch, 1);
      }
    }
  }
}
 
// --------------------------------------------------------------------------
// Thread receives packets from the serial port and sends them out the USB
// It uses a buffered stdio input, an un-buffered low-level file output.
// This isn't terribly efficient, but rather an example of both methods.
 
void ser2usb_func(cyg_addrword_t data)
{
  int  c;
  FILE *rxf = fopen(SER_DEV, "r");
  int  txh = open(USB_TX_DEV, O_WRONLY, 0);
 
  DBG("Ser2Usb: Thread starting\n");
 
  if (!rxf) {
    DBG("Error opening serial rx port\n");
    return;
  }
 
  if (txh < 0) {
    DBG("Error opening USB tx port\n");
    return;
  }
 
  while (1) {
 
    // ----- Wait for the host to configure -----
 
    DBG("Ser2Usb: Waiting for USB configuration\n");
    usbs_serial_wait_until_configured();
    cyg_thread_delay((cyg_tick_count_t) 10);
 
    // ----- While configured read data & send out serial port -----
 
    DBG("Ser2Usb: USB configured\n");
    while (usbs_serial_is_configured()) {
      if ((c = getc(rxf)) < 0) {
        DBG("*** Console Read Error: %d ***\n", c);
      }
      else {
        char ch = (char) c;
        write(txh, &ch, 1);
      }
    }
  }
}
 
// --------------------------------------------------------------------------
//  Application Startup
// --------------------------------------------------------------------------
 
void cyg_user_start(void)
{
  DBG("Entering cyg_user_start() function\n");
 
#ifdef CYGDAT_IO_USB_SLAVE_CLASS_TYPE_ACM
  // Override the class handler to use ours.
  usbs_serial_ep0->class_control_fn = acm_class_handler;
#endif
 
  cyg_thread_create(4, usb2ser_func, (cyg_addrword_t) 0,
                    "Usb2Serial", (void *) stack[0], THREAD_STACK_SIZE,
                    &usb2ser_thread, &thread[0]);
 
  cyg_thread_create(4, ser2usb_func, (cyg_addrword_t) 1,
                    "Serial2Usb", (void *) stack[1], THREAD_STACK_SIZE,
                    &ser2usb_thread, &thread[1]);
 
  // Start USB subsystem
  usbs_serial_start();
 
  // Start the threads running.
  cyg_thread_resume(usb2ser_thread);
  cyg_thread_resume(ser2usb_thread);
}
 
 

Go to most recent revision | Compare with Previous | Blame | View Log

powered by: WebSVN 2.1.0

© copyright 1999-2025 OpenCores.org, equivalent to Oliscience, all rights reserved. OpenCores®, registered trademark.