Home | History | Annotate | Download | only in netmgr
      1 /*
      2  * Copyright 2018, 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 "fork.h"
     18 
     19 #include "log.h"
     20 
     21 #include <errno.h>
     22 #include <string.h>
     23 #include <sys/types.h>
     24 #include <sys/wait.h>
     25 #include <unistd.h>
     26 
     27 bool forkAndExec(const char* argv[]) {
     28     pid_t pid = ::fork();
     29     if (pid < 0) {
     30         // Failed to fork
     31         LOGE("fork() failed: %s", strerror(errno));
     32         return false;
     33     } else if (pid == 0) {
     34         // Child
     35         char buffer[32768];
     36         size_t offset = 0;
     37         for (size_t i = 0; argv[i]; ++i) {
     38             offset += snprintf(buffer + offset, sizeof(buffer) - offset,
     39                                "%s ", argv[i]);
     40         }
     41         LOGE("Running '%s'", buffer);
     42         execvp(argv[0], const_cast<char* const*>(argv));
     43         LOGE("Failed to run '%s': %s", argv[0], strerror(errno));
     44         _exit(1);
     45     } else {
     46         // Parent
     47         int status = 0;
     48         do {
     49             ::waitpid(pid, &status, 0);
     50             if (WIFEXITED(status)) {
     51                 int exitStatus = WEXITSTATUS(status);
     52                 if (exitStatus == 0) {
     53                     return true;
     54                 }
     55                 LOGE("Error: '%s' exited with code: %d", argv[0], exitStatus);
     56             } else if (WIFSIGNALED(status)) {
     57                 LOGE("Error: '%s' terminated with signal: %d",
     58                      argv[0], WTERMSIG(status));
     59             }
     60             // Other possibilities include being stopped and continued as part
     61             // of a trace but we don't really care about that. The important
     62             // part is that unless the process explicitly exited or was killed
     63             // by a signal we have to keep waiting.
     64         } while (!WIFEXITED(status) && !WIFSIGNALED(status));
     65         return false;
     66     }
     67 }
     68