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 }
1.5.3