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
fb_get_error(void)46 char *fb_get_error(void)
47 {
48 return ERROR;
49 }
50
check_response(Transport * transport,uint32_t size,char * response)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
_command_start(Transport * transport,const char * cmd,uint32_t size,char * response)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
_command_data(Transport * transport,const void * data,uint32_t size)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
_command_end(Transport * transport)143 static int _command_end(Transport* transport) {
144 return check_response(transport, 0, 0) < 0 ? -1 : 0;
145 }
146
_command_send(Transport * transport,const char * cmd,const void * data,uint32_t size,char * response)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
_command_send_no_data(Transport * transport,const char * cmd,char * response)171 static int _command_send_no_data(Transport* transport, const char* cmd, char* response) {
172 return _command_start(transport, cmd, 0, response);
173 }
174
fb_command(Transport * transport,const char * cmd)175 int fb_command(Transport* transport, const char* cmd) {
176 return _command_send_no_data(transport, cmd, 0);
177 }
178
fb_command_response(Transport * transport,const char * cmd,char * response)179 int fb_command_response(Transport* transport, const char* cmd, char* response) {
180 return _command_send_no_data(transport, cmd, response);
181 }
182
fb_download_data(Transport * transport,const void * data,uint32_t size)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
fb_download_data_sparse_write(void * priv,const void * data,int len)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
fb_download_data_sparse_flush(Transport * transport)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
fb_download_data_sparse(Transport * transport,struct sparse_file * s)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