storage_task.c

Go to the documentation of this file.
00001 /*This file has been prepared for Doxygen automatic documentation generation.*/
00013 
00014 /* Copyright (c) 2007, Atmel Corporation All rights reserved.
00015  *
00016  * Redistribution and use in source and binary forms, with or without
00017  * modification, are permitted provided that the following conditions are met:
00018  *
00019  * 1. Redistributions of source code must retain the above copyright notice,
00020  * this list of conditions and the following disclaimer.
00021  *
00022  * 2. Redistributions in binary form must reproduce the above copyright notice,
00023  * this list of conditions and the following disclaimer in the documentation
00024  * and/or other materials provided with the distribution.
00025  *
00026  * 3. The name of ATMEL may not be used to endorse or promote products derived
00027  * from this software without specific prior written permission.
00028  *
00029  * THIS SOFTWARE IS PROVIDED BY ATMEL ``AS IS'' AND ANY EXPRESS OR IMPLIED
00030  * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
00031  * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE EXPRESSLY AND
00032  * SPECIFICALLY DISCLAIMED. IN NO EVENT SHALL ATMEL BE LIABLE FOR ANY DIRECT,
00033  * INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
00034  * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
00035  * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
00036  * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
00037  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
00038  * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
00039  */
00040 
00041 //_____  I N C L U D E S ___________________________________________________
00042 
00043 #include "config.h"
00044 #include "conf_usb.h"
00045 #include "storage_task.h"
00046 #include "lib_mcu/usb/usb_drv.h"
00047 #include "usb_descriptors.h"
00048 #include "modules/usb/device_chap9/usb_standard_request.h"
00049 #include "usb_specific_request.h"
00050 #include "modules/scsi_decoder/scsi_decoder.h"
00051 #include "modules/control_access/ctrl_access.h"
00052 #include "lib_mem/mmc_sd/mmc_sd.h"
00053 
00054 
00055 //_____ D E C L A R A T I O N S ____________________________________________
00056 
00057 
00058 
00059 bit ms_data_direction;
00060 static _MEM_TYPE_SLOW_ U8  dCBWTag[4];
00061 
00062 extern _MEM_TYPE_SLOW_ U8  g_scsi_status;
00063 extern _MEM_TYPE_FAST_ U32 g_scsi_data_remaining;
00064 extern bit ms_multiple_drive;
00065 extern _MEM_TYPE_SLOW_ U8  g_scsi_command[16];
00066 
00067 extern U8   endpoint_status[MAX_EP_NB];
00068 
00069 _MEM_TYPE_SLOW_ U8 usb_LUN;
00070 
00071 
00072 void usb_mass_storage_cbw (void);
00073 void usb_mass_storage_csw (void);
00074 
00084 void storage_task_init(void)
00085 {
00086    Leds_init();
00087    Usb_enable_sof_interrupt();
00088    df_mem_init();    // Init the hw/sw ressources required to drive the DF.
00089 }
00090 
00091 
00100 void storage_task(void)
00101 {
00102    if (Is_device_enumerated())
00103    {
00104       Usb_select_endpoint(EP_MS_OUT);
00105       if (Is_usb_receive_out())
00106       {
00107          usb_mass_storage_cbw();
00108          usb_mass_storage_csw();
00109       }
00110    }
00111 }
00112 
00113 
00114 
00115 
00116 
00117 
00127 void usb_mass_storage_cbw (void)
00128 {
00129 bit cbw_error;
00130 U8  c;
00131 U8  dummy;
00132 
00133    cbw_error = FALSE;
00134    Usb_select_endpoint(EP_MS_OUT);           
00135    if (0x55 != Usb_read_byte())
00136       { cbw_error = TRUE; } 
00137    if (0x53 != Usb_read_byte())
00138       { cbw_error = TRUE; } 
00139    if (0x42 != Usb_read_byte())
00140       { cbw_error = TRUE; } 
00141    if (0x43 != Usb_read_byte())
00142       { cbw_error = TRUE; } 
00143    if (cbw_error)
00144    {
00145       Usb_ack_receive_out();
00146       Usb_select_endpoint(EP_MS_IN);
00147       Usb_enable_stall_handshake();
00148       endpoint_status[(EP_MS_IN & MSK_EP_DIR)] = 0x01;
00149       return;
00150    }
00151 
00152    dCBWTag[0] = Usb_read_byte();             
00153    dCBWTag[1] = Usb_read_byte();
00154    dCBWTag[2] = Usb_read_byte();
00155    dCBWTag[3] = Usb_read_byte();
00156    
00157    LSB0(g_scsi_data_remaining) = Usb_read_byte();
00158    LSB1(g_scsi_data_remaining) = Usb_read_byte();
00159    LSB2(g_scsi_data_remaining) = Usb_read_byte();
00160    LSB3(g_scsi_data_remaining) = Usb_read_byte();
00161 
00162    if (Usb_read_byte() != 0x00)              
00163    {
00164       Usb_set_ms_data_direction_in();
00165    }
00166    else
00167    {
00168       Usb_set_ms_data_direction_out();
00169    }
00170 
00171    usb_LUN = Usb_read_byte();
00172 
00173    if (!ms_multiple_drive)
00174    {
00175       usb_LUN = get_cur_lun();
00176    }
00177 
00178    dummy      = Usb_read_byte();                // dummy CBWCBLength read
00179 
00180 
00181    for (c=0; c<16; c++)                         // store scsi_command
00182    {
00183       g_scsi_command[c] = Usb_read_byte();
00184    }
00185    Usb_ack_receive_out();
00186 
00187    if (Is_usb_ms_data_direction_in())
00188    {
00189       Usb_select_endpoint(EP_MS_IN);
00190    }
00191 
00192    if (TRUE != scsi_decode_command())
00193    {
00194       U8 ep;
00195       Usb_enable_stall_handshake();
00196       if (Is_usb_ms_data_direction_in())
00197       {
00198          ep = (EP_MS_IN & MSK_EP_DIR);
00199       }else{
00200          ep = (EP_MS_OUT & MSK_EP_DIR);
00201       }
00202       endpoint_status[ep] = 0x01;
00203    }
00204 }
00205 
00206 
00215 void usb_mass_storage_csw (void)
00216 {
00217    Usb_select_endpoint(EP_MS_IN);
00218    while (Is_usb_endpoint_stall_requested())
00219    {
00220       Usb_select_endpoint(EP_CONTROL);
00221       if (Is_usb_receive_setup())       { usb_process_request(); }
00222       Usb_select_endpoint(EP_MS_IN);
00223    }
00224 
00225    Usb_select_endpoint(EP_MS_OUT);
00226    while (Is_usb_endpoint_stall_requested())
00227    {
00228       Usb_select_endpoint(EP_CONTROL);
00229       if (Is_usb_receive_setup())       { usb_process_request(); }
00230       Usb_select_endpoint(EP_MS_OUT);
00231    }
00232 
00233 
00234    Usb_select_endpoint(EP_MS_IN);
00235    while(!Is_usb_write_enabled())
00236    {
00237       if(!Is_usb_endpoint_enabled())   return; // USB Reset
00238    }
00240    Usb_write_byte(0x55);                                 
00241    Usb_write_byte(0x53);                                 
00242    Usb_write_byte(0x42);                                 
00243    Usb_write_byte(0x53);                                 
00244 
00245    Usb_write_byte(dCBWTag[0]);
00246    Usb_write_byte(dCBWTag[1]);
00247    Usb_write_byte(dCBWTag[2]);
00248    Usb_write_byte(dCBWTag[3]);
00250    Usb_write_byte( LSB0(g_scsi_data_remaining) );
00251    Usb_write_byte( LSB1(g_scsi_data_remaining) );
00252    Usb_write_byte( LSB2(g_scsi_data_remaining) );
00253    Usb_write_byte( LSB3(g_scsi_data_remaining) );
00254 
00256    Usb_write_byte(g_scsi_status);                        
00257 
00258    Usb_send_in();
00259 }

Generated on Fri Oct 31 14:31:24 2008 for ATMEL by  doxygen 1.5.3