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 round_down(a, b) \ 30 ({ typeof(a) _a = (a); typeof(b) _b = (b); _a - (_a % _b); }) 31 32 #include <stdio.h> 33 #include <stdlib.h> 34 #include <string.h> 35 #include <errno.h> 36 37 #include <algorithm> 38 39 #include <sparse/sparse.h> 40 41 #include "fastboot.h" 42 #include "transport.h" 43 44 static char ERROR[128]; 45 46 char *fb_get_error(void) 47 { 48 return ERROR; 49 } 50 51 static int check_response(Transport* transport, uint32_t size, char* response) { 52 char status[65]; 53 54 while (true) { 55 int r = transport->Read(status, 64); 56 if (r < 0) { 57 sprintf(ERROR, "status read failed (%s)", strerror(errno)); 58 transport->Close(); 59 return -1; 60 } 61 status[r] = 0; 62 63 if (r < 4) { 64 sprintf(ERROR, "status malformed (%d bytes)", r); 65 transport->Close(); 66 return -1; 67 } 68 69 if (!memcmp(status, "INFO", 4)) { 70 fprintf(stderr,"(bootloader) %s\n", status + 4); 71 continue; 72 } 73 74 if (!memcmp(status, "OKAY", 4)) { 75 if (response) { 76 strcpy(response, (char*) status + 4); 77 } 78 return 0; 79 } 80 81 if (!memcmp(status, "FAIL", 4)) { 82 if (r > 4) { 83 sprintf(ERROR, "remote: %s", status + 4); 84 } else { 85 strcpy(ERROR, "remote failure"); 86 } 87 return -1; 88 } 89 90 if (!memcmp(status, "DATA", 4) && size > 0){ 91 uint32_t dsize = strtol(status + 4, 0, 16); 92 if (dsize > size) { 93 strcpy(ERROR, "data size too large"); 94 transport->Close(); 95 return -1; 96 } 97 return dsize; 98 } 99 100 strcpy(ERROR,"unknown status code"); 101 transport->Close(); 102 break; 103 } 104 105 return -1; 106 } 107 108 static int _command_start(Transport* transport, const char* cmd, uint32_t size, char* response) { 109 size_t cmdsize = strlen(cmd); 110 if (cmdsize > 64) { 111 sprintf(ERROR, "command too large"); 112 return -1; 113 } 114 115 if (response) { 116 response[0] = 0; 117 } 118 119 if (transport->Write(cmd, cmdsize) != static_cast<int>(cmdsize)) { 120 sprintf(ERROR, "command write failed (%s)", strerror(errno)); 121 transport->Close(); 122 return -1; 123 } 124 125 return check_response(transport, size, response); 126 } 127 128 static int _command_data(Transport* transport, const void* data, uint32_t size) { 129 int r = transport->Write(data, size); 130 if (r < 0) { 131 sprintf(ERROR, "data transfer failure (%s)", strerror(errno)); 132 transport->Close(); 133 return -1; 134 } 135 if (r != ((int) size)) { 136 sprintf(ERROR, "data transfer failure (short transfer)"); 137 transport->Close(); 138 return -1; 139 } 140 return r; 141 } 142 143 static int _command_end(Transport* transport) { 144 return check_response(transport, 0, 0) < 0 ? -1 : 0; 145 } 146 147 static int _command_send(Transport* transport, const char* cmd, const void* data, uint32_t size, 148 char* response) { 149 if (size == 0) { 150 return -1; 151 } 152 153 int r = _command_start(transport, cmd, size, response); 154 if (r < 0) { 155 return -1; 156 } 157 158 r = _command_data(transport, data, size); 159 if (r < 0) { 160 return -1; 161 } 162 163 r = _command_end(transport); 164 if (r < 0) { 165 return -1; 166 } 167 168 return size; 169 } 170 171 static int _command_send_no_data(Transport* transport, const char* cmd, char* response) { 172 return _command_start(transport, cmd, 0, response); 173 } 174 175 int fb_command(Transport* transport, const char* cmd) { 176 return _command_send_no_data(transport, cmd, 0); 177 } 178 179 int fb_command_response(Transport* transport, const char* cmd, char* response) { 180 return _command_send_no_data(transport, cmd, response); 181 } 182 183 int fb_download_data(Transport* transport, const void* data, uint32_t size) { 184 char cmd[64]; 185 sprintf(cmd, "download:%08x", size); 186 return _command_send(transport, cmd, data, size, 0) < 0 ? -1 : 0; 187 } 188 189 #define TRANSPORT_BUF_SIZE 1024 190 static char transport_buf[TRANSPORT_BUF_SIZE]; 191 static int transport_buf_len; 192 193 static int fb_download_data_sparse_write(void *priv, const void *data, int len) 194 { 195 int r; 196 Transport* transport = reinterpret_cast<Transport*>(priv); 197 int to_write; 198 const char* ptr = reinterpret_cast<const char*>(data); 199 200 if (transport_buf_len) { 201 to_write = std::min(TRANSPORT_BUF_SIZE - transport_buf_len, len); 202 203 memcpy(transport_buf + transport_buf_len, ptr, to_write); 204 transport_buf_len += to_write; 205 ptr += to_write; 206 len -= to_write; 207 } 208 209 if (transport_buf_len == TRANSPORT_BUF_SIZE) { 210 r = _command_data(transport, transport_buf, TRANSPORT_BUF_SIZE); 211 if (r != TRANSPORT_BUF_SIZE) { 212 return -1; 213 } 214 transport_buf_len = 0; 215 } 216 217 if (len > TRANSPORT_BUF_SIZE) { 218 if (transport_buf_len > 0) { 219 sprintf(ERROR, "internal error: transport_buf not empty\n"); 220 return -1; 221 } 222 to_write = round_down(len, TRANSPORT_BUF_SIZE); 223 r = _command_data(transport, ptr, to_write); 224 if (r != to_write) { 225 return -1; 226 } 227 ptr += to_write; 228 len -= to_write; 229 } 230 231 if (len > 0) { 232 if (len > TRANSPORT_BUF_SIZE) { 233 sprintf(ERROR, "internal error: too much left for transport_buf\n"); 234 return -1; 235 } 236 memcpy(transport_buf, ptr, len); 237 transport_buf_len = len; 238 } 239 240 return 0; 241 } 242 243 static int fb_download_data_sparse_flush(Transport* transport) { 244 if (transport_buf_len > 0) { 245 if (_command_data(transport, transport_buf, transport_buf_len) != transport_buf_len) { 246 return -1; 247 } 248 transport_buf_len = 0; 249 } 250 return 0; 251 } 252 253 int fb_download_data_sparse(Transport* transport, struct sparse_file* s) { 254 int size = sparse_file_len(s, true, false); 255 if (size <= 0) { 256 return -1; 257 } 258 259 char cmd[64]; 260 sprintf(cmd, "download:%08x", size); 261 int r = _command_start(transport, cmd, size, 0); 262 if (r < 0) { 263 return -1; 264 } 265 266 r = sparse_file_callback(s, true, false, fb_download_data_sparse_write, transport); 267 if (r < 0) { 268 return -1; 269 } 270 271 r = fb_download_data_sparse_flush(transport); 272 if (r < 0) { 273 return -1; 274 } 275 276 return _command_end(transport); 277 } 278