blob: 654e0f32d299ad8e32132060c9481513bf0e9357 [file] [log] [blame]
Christopher Tated2f54152011-04-21 12:53:28 -07001/*
2 * Copyright (C) 2011 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17#include <unistd.h>
18#include <stdio.h>
19
20#include "sysdeps.h"
21
22#define TRACE_TAG TRACE_ADB
23#include "adb.h"
24
Christopher Tate10f129c2011-06-21 15:59:25 -070025typedef struct {
26 pid_t pid;
27 int fd;
28} backup_harvest_params;
29
Christopher Tate5b811fa2011-06-10 11:38:37 -070030// socketpair but do *not* mark as close_on_exec
31static int backup_socketpair(int sv[2]) {
32 int rc = unix_socketpair( AF_UNIX, SOCK_STREAM, 0, sv );
33 if (rc < 0)
34 return -1;
35
36 return 0;
37}
38
Christopher Tate10f129c2011-06-21 15:59:25 -070039// harvest the child process then close the read end of the socketpair
40static void* backup_child_waiter(void* args) {
Christopher Tate5b811fa2011-06-10 11:38:37 -070041 int status;
Christopher Tate10f129c2011-06-21 15:59:25 -070042 backup_harvest_params* params = (backup_harvest_params*) args;
Christopher Tate5b811fa2011-06-10 11:38:37 -070043
Christopher Tate10f129c2011-06-21 15:59:25 -070044 waitpid(params->pid, &status, 0);
45 adb_close(params->fd);
46 free(params);
Christopher Tate5b811fa2011-06-10 11:38:37 -070047 return NULL;
48}
49
Christopher Tated2f54152011-04-21 12:53:28 -070050/* returns the data socket passing the backup data here for forwarding */
Christopher Tate702967a2011-05-17 15:52:54 -070051int backup_service(BackupOperation op, char* args) {
Christopher Tated2f54152011-04-21 12:53:28 -070052 pid_t pid;
53 int s[2];
Christopher Tate702967a2011-05-17 15:52:54 -070054 char* operation;
Christopher Tated2f54152011-04-21 12:53:28 -070055
Mark Salyzyn60299df2014-04-30 09:10:31 -070056 // Command string depends on our invocation
Christopher Tate702967a2011-05-17 15:52:54 -070057 if (op == BACKUP) {
58 operation = "backup";
Christopher Tate702967a2011-05-17 15:52:54 -070059 } else {
60 operation = "restore";
Christopher Tate702967a2011-05-17 15:52:54 -070061 }
62
63 D("backup_service(%s, %s)\n", operation, args);
Christopher Tated2f54152011-04-21 12:53:28 -070064
65 // set up the pipe from the subprocess to here
66 // parent will read s[0]; child will write s[1]
Christopher Tate5b811fa2011-06-10 11:38:37 -070067 if (backup_socketpair(s)) {
Christopher Tate702967a2011-05-17 15:52:54 -070068 D("can't create backup/restore socketpair\n");
69 fprintf(stderr, "unable to create backup/restore socketpair\n");
Christopher Tated2f54152011-04-21 12:53:28 -070070 return -1;
71 }
72
Christopher Tate5b811fa2011-06-10 11:38:37 -070073 D("Backup/restore socket pair: (send=%d, receive=%d)\n", s[1], s[0]);
74 close_on_exec(s[0]); // only the side we hold on to
75
Christopher Tated2f54152011-04-21 12:53:28 -070076 // spin off the child process to run the backup command
77 pid = fork();
78 if (pid < 0) {
79 // failure
Christopher Tate702967a2011-05-17 15:52:54 -070080 D("can't fork for %s\n", operation);
81 fprintf(stderr, "unable to fork for %s\n", operation);
Christopher Tated2f54152011-04-21 12:53:28 -070082 adb_close(s[0]);
83 adb_close(s[1]);
84 return -1;
85 }
86
87 // Great, we're off and running.
88 if (pid == 0) {
Christopher Tate5b811fa2011-06-10 11:38:37 -070089 // child -- actually run the backup here
Christopher Tated2f54152011-04-21 12:53:28 -070090 char* p;
91 int argc;
Christopher Tate5b811fa2011-06-10 11:38:37 -070092 char portnum[16];
Christopher Tate702967a2011-05-17 15:52:54 -070093 char** bu_args;
Christopher Tated2f54152011-04-21 12:53:28 -070094
Christopher Tate5b811fa2011-06-10 11:38:37 -070095 // fixed args: [0] is 'bu', [1] is the port number, [2] is the 'operation' string
96 argc = 3;
Christopher Tated2f54152011-04-21 12:53:28 -070097 for (p = (char*)args; p && *p; ) {
98 argc++;
99 while (*p && *p != ':') p++;
100 if (*p == ':') p++;
101 }
102
Christopher Tate702967a2011-05-17 15:52:54 -0700103 bu_args = (char**) alloca(argc*sizeof(char*) + 1);
Christopher Tate5b811fa2011-06-10 11:38:37 -0700104
105 // run through again to build the argv array
106 argc = 0;
107 bu_args[argc++] = "bu";
108 snprintf(portnum, sizeof(portnum), "%d", s[1]);
109 bu_args[argc++] = portnum;
110 bu_args[argc++] = operation;
Christopher Tate702967a2011-05-17 15:52:54 -0700111 for (p = (char*)args; p && *p; ) {
112 bu_args[argc++] = p;
Christopher Tated2f54152011-04-21 12:53:28 -0700113 while (*p && *p != ':') p++;
114 if (*p == ':') {
115 *p = 0;
116 p++;
117 }
118 }
Christopher Tate702967a2011-05-17 15:52:54 -0700119 bu_args[argc] = NULL;
Christopher Tated2f54152011-04-21 12:53:28 -0700120
121 // Close the half of the socket that we don't care about, route 'bu's console
122 // to the output socket, and off we go
123 adb_close(s[0]);
Christopher Tated2f54152011-04-21 12:53:28 -0700124
125 // off we go
Christopher Tate702967a2011-05-17 15:52:54 -0700126 execvp("/system/bin/bu", (char * const *)bu_args);
Christopher Tated2f54152011-04-21 12:53:28 -0700127 // oops error - close up shop and go home
128 fprintf(stderr, "Unable to exec 'bu', bailing\n");
129 exit(-1);
130 } else {
Christopher Tate5b811fa2011-06-10 11:38:37 -0700131 adb_thread_t t;
Christopher Tate10f129c2011-06-21 15:59:25 -0700132 backup_harvest_params* params;
Christopher Tate5b811fa2011-06-10 11:38:37 -0700133
Christopher Tated2f54152011-04-21 12:53:28 -0700134 // parent, i.e. adbd -- close the sending half of the socket
Christopher Tate5b811fa2011-06-10 11:38:37 -0700135 D("fork() returned pid %d\n", pid);
Christopher Tated2f54152011-04-21 12:53:28 -0700136 adb_close(s[1]);
Christopher Tate5b811fa2011-06-10 11:38:37 -0700137
138 // spin a thread to harvest the child process
Christopher Tate10f129c2011-06-21 15:59:25 -0700139 params = (backup_harvest_params*) malloc(sizeof(backup_harvest_params));
140 params->pid = pid;
141 params->fd = s[0];
142 if (adb_thread_create(&t, backup_child_waiter, params)) {
Christopher Tate5b811fa2011-06-10 11:38:37 -0700143 adb_close(s[0]);
Christopher Tate10f129c2011-06-21 15:59:25 -0700144 free(params);
Christopher Tate5b811fa2011-06-10 11:38:37 -0700145 D("Unable to create child harvester\n");
146 return -1;
147 }
Christopher Tated2f54152011-04-21 12:53:28 -0700148 }
149
150 // we'll be reading from s[0] as the data is sent by the child process
151 return s[0];
152}