1 /* 2 * Copyright (C) 2008 The Android Open Source Project 3 * All rights reserved. 4 * 5 * Redistribution and use in source and binary forms, with or without 6 * modification, are permitted provided that the following conditions 7 * are met: 8 * * Redistributions of source code must retain the above copyright 9 * notice, this list of conditions and the following disclaimer. 10 * * Redistributions in binary form must reproduce the above copyright 11 * notice, this list of conditions and the following disclaimer in 12 * the documentation and/or other materials provided with the 13 * distribution. 14 * 15 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS 16 * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT 17 * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS 18 * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE 19 * COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, 20 * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, 21 * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS 22 * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED 23 * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, 24 * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT 25 * OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 26 * SUCH DAMAGE. 27 */ 28 29 #define min(a, b) \ 30 ({ typeof(a) _a = (a); typeof(b) _b = (b); (_a < _b) ? _a : _b; }) 31 #define round_down(a, b) \ 32 ({ typeof(a) _a = (a); typeof(b) _b = (b); _a - (_a % _b); }) 33 34 #include <stdio.h> 35 #include <stdlib.h> 36 #include <string.h> 37 #include <errno.h> 38 39 #include <sparse/sparse.h> 40 41 #include "fastboot.h" 42 43 static char ERROR[128]; 44 45 char *fb_get_error(void) 46 { 47 return ERROR; 48 } 49 50 static int check_response(usb_handle *usb, unsigned int size, char *response) 51 { 52 unsigned char status[65]; 53 int r; 54 55 for(;;) { 56 r = usb_read(usb, status, 64); 57 if(r < 0) { 58 sprintf(ERROR, "status read failed (%s)", strerror(errno)); 59 usb_close(usb); 60 return -1; 61 } 62 status[r] = 0; 63 64 if(r < 4) { 65 sprintf(ERROR, "status malformed (%d bytes)", r); 66 usb_close(usb); 67 return -1; 68 } 69 70 if(!memcmp(status, "INFO", 4)) { 71 fprintf(stderr,"(bootloader) %s\n", status + 4); 72 continue; 73 } 74 75 if(!memcmp(status, "OKAY", 4)) { 76 if(response) { 77 strcpy(response, (char*) status + 4); 78 } 79 return 0; 80 } 81 82 if(!memcmp(status, "FAIL", 4)) { 83 if(r > 4) { 84 sprintf(ERROR, "remote: %s", status + 4); 85 } else { 86 strcpy(ERROR, "remote failure"); 87 } 88 return -1; 89 } 90 91 if(!memcmp(status, "DATA", 4) && size > 0){ 92 unsigned dsize = strtoul((char*) status + 4, 0, 16); 93 if(dsize > size) { 94 strcpy(ERROR, "data size too large"); 95 usb_close(usb); 96 return -1; 97 } 98 return dsize; 99 } 100 101 strcpy(ERROR,"unknown status code"); 102 usb_close(usb); 103 break; 104 } 105 106 return -1; 107 } 108 109 static int _command_start(usb_handle *usb, const char *cmd, unsigned size, 110 char *response) 111 { 112 int cmdsize = strlen(cmd); 113 int r; 114 115 if(response) { 116 response[0] = 0; 117 } 118 119 if(cmdsize > 64) { 120 sprintf(ERROR,"command too large"); 121 return -1; 122 } 123 124 if(usb_write(usb, cmd, cmdsize) != cmdsize) { 125 sprintf(ERROR,"command write failed (%s)", strerror(errno)); 126 usb_close(usb); 127 return -1; 128 } 129 130 return check_response(usb, size, response); 131 } 132 133 static int _command_data(usb_handle *usb, const void *data, unsigned size) 134 { 135 int r; 136 137 r = usb_write(usb, data, size); 138 if(r < 0) { 139 sprintf(ERROR, "data transfer failure (%s)", strerror(errno)); 140 usb_close(usb); 141 return -1; 142 } 143 if(r != ((int) size)) { 144 sprintf(ERROR, "data transfer failure (short transfer)"); 145 usb_close(usb); 146 return -1; 147 } 148 149 return r; 150 } 151 152 static int _command_end(usb_handle *usb) 153 { 154 int r; 155 r = check_response(usb, 0, 0); 156 if(r < 0) { 157 return -1; 158 } 159 return 0; 160 } 161 162 static int _command_send(usb_handle *usb, const char *cmd, 163 const void *data, unsigned size, 164 char *response) 165 { 166 int r; 167 if (size == 0) { 168 return -1; 169 } 170 171 r = _command_start(usb, cmd, size, response); 172 if (r < 0) { 173 return -1; 174 } 175 176 r = _command_data(usb, data, size); 177 if (r < 0) { 178 return -1; 179 } 180 181 r = _command_end(usb); 182 if(r < 0) { 183 return -1; 184 } 185 186 return size; 187 } 188 189 static int _command_send_no_data(usb_handle *usb, const char *cmd, 190 char *response) 191 { 192 int r; 193 194 return _command_start(usb, cmd, 0, response); 195 } 196 197 int fb_command(usb_handle *usb, const char *cmd) 198 { 199 return _command_send_no_data(usb, cmd, 0); 200 } 201 202 int fb_command_response(usb_handle *usb, const char *cmd, char *response) 203 { 204 return _command_send_no_data(usb, cmd, response); 205 } 206 207 int fb_download_data(usb_handle *usb, const void *data, unsigned size) 208 { 209 char cmd[64]; 210 int r; 211 212 sprintf(cmd, "download:%08x", size); 213 r = _command_send(usb, cmd, data, size, 0); 214 215 if(r < 0) { 216 return -1; 217 } else { 218 return 0; 219 } 220 } 221 222 #define USB_BUF_SIZE 512 223 static char usb_buf[USB_BUF_SIZE]; 224 static int usb_buf_len; 225 226 static int fb_download_data_sparse_write(void *priv, const void *data, int len) 227 { 228 int r; 229 usb_handle *usb = priv; 230 int to_write; 231 const char *ptr = data; 232 233 if (usb_buf_len) { 234 to_write = min(USB_BUF_SIZE - usb_buf_len, len); 235 236 memcpy(usb_buf + usb_buf_len, ptr, to_write); 237 usb_buf_len += to_write; 238 ptr += to_write; 239 len -= to_write; 240 } 241 242 if (usb_buf_len == USB_BUF_SIZE) { 243 r = _command_data(usb, usb_buf, USB_BUF_SIZE); 244 if (r != USB_BUF_SIZE) { 245 return -1; 246 } 247 usb_buf_len = 0; 248 } 249 250 if (len > USB_BUF_SIZE) { 251 if (usb_buf_len > 0) { 252 sprintf(ERROR, "internal error: usb_buf not empty\n"); 253 return -1; 254 } 255 to_write = round_down(len, USB_BUF_SIZE); 256 r = _command_data(usb, ptr, to_write); 257 if (r != to_write) { 258 return -1; 259 } 260 ptr += to_write; 261 len -= to_write; 262 } 263 264 if (len > 0) { 265 if (len > USB_BUF_SIZE) { 266 sprintf(ERROR, "internal error: too much left for usb_buf\n"); 267 return -1; 268 } 269 memcpy(usb_buf, ptr, len); 270 usb_buf_len = len; 271 } 272 273 return 0; 274 } 275 276 static int fb_download_data_sparse_flush(usb_handle *usb) 277 { 278 int r; 279 280 if (usb_buf_len > 0) { 281 r = _command_data(usb, usb_buf, usb_buf_len); 282 if (r != usb_buf_len) { 283 return -1; 284 } 285 usb_buf_len = 0; 286 } 287 288 return 0; 289 } 290 291 int fb_download_data_sparse(usb_handle *usb, struct sparse_file *s) 292 { 293 char cmd[64]; 294 int r; 295 int size = sparse_file_len(s, true, false); 296 if (size <= 0) { 297 return -1; 298 } 299 300 sprintf(cmd, "download:%08x", size); 301 r = _command_start(usb, cmd, size, 0); 302 if (r < 0) { 303 return -1; 304 } 305 306 r = sparse_file_callback(s, true, false, fb_download_data_sparse_write, usb); 307 if (r < 0) { 308 return -1; 309 } 310 311 fb_download_data_sparse_flush(usb); 312 313 return _command_end(usb); 314 } 315