chromium/third_party/crashpad/crashpad/util/mac/service_management.cc

// Copyright 2014 The Crashpad Authors
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
//     http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.

#include "util/mac/service_management.h"

#include <errno.h>
#include <launch.h>

#include "base/mac/scoped_launch_data.h"
#include "util/mac/launchd.h"
#include "util/misc/clock.h"

namespace crashpad {

namespace {

launch_data_t LaunchDataDictionaryForJob(const std::string& label) {
  base::mac::ScopedLaunchData request(LaunchDataAlloc(LAUNCH_DATA_DICTIONARY));
  LaunchDataDictInsert(
      request.get(), LaunchDataNewString(label.c_str()), LAUNCH_KEY_GETJOB);

  base::mac::ScopedLaunchData response(LaunchMsg(request.get()));
  if (LaunchDataGetType(response.get()) != LAUNCH_DATA_DICTIONARY) {
    return nullptr;
  }

  return response.release();
}

}  // namespace

bool ServiceManagementSubmitJob(CFDictionaryRef job_cf) {
  base::mac::ScopedLaunchData job_launch(CFPropertyToLaunchData(job_cf));
  if (!job_launch.get()) {
    return false;
  }

  base::mac::ScopedLaunchData jobs(LaunchDataAlloc(LAUNCH_DATA_ARRAY));
  LaunchDataArraySetIndex(jobs.get(), job_launch.release(), 0);

  base::mac::ScopedLaunchData request(LaunchDataAlloc(LAUNCH_DATA_DICTIONARY));
  LaunchDataDictInsert(request.get(), jobs.release(), LAUNCH_KEY_SUBMITJOB);

  base::mac::ScopedLaunchData response(LaunchMsg(request.get()));
  if (LaunchDataGetType(response.get()) != LAUNCH_DATA_ARRAY) {
    return false;
  }

  if (LaunchDataArrayGetCount(response.get()) != 1) {
    return false;
  }

  launch_data_t response_element = LaunchDataArrayGetIndex(response.get(), 0);
  if (LaunchDataGetType(response_element) != LAUNCH_DATA_ERRNO) {
    return false;
  }

  int err = LaunchDataGetErrno(response_element);
  if (err != 0) {
    return false;
  }

  return true;
}

bool ServiceManagementRemoveJob(const std::string& label, bool wait) {
  base::mac::ScopedLaunchData request(LaunchDataAlloc(LAUNCH_DATA_DICTIONARY));
  LaunchDataDictInsert(
      request.get(), LaunchDataNewString(label.c_str()), LAUNCH_KEY_REMOVEJOB);

  base::mac::ScopedLaunchData response(LaunchMsg(request.get()));
  if (LaunchDataGetType(response.get()) != LAUNCH_DATA_ERRNO) {
    return false;
  }

  int err = LaunchDataGetErrno(response.get());
  if (err == EINPROGRESS) {
    if (wait) {
      // TODO(mark): Use a kqueue to wait for the process to exit. To avoid a
      // race, the kqueue would need to be set up prior to asking launchd to
      // remove the job. Even so, the job’s PID may change between the time it’s
      // obtained and the time the kqueue is set up, so this is nontrivial.
      do {
        SleepNanoseconds(1E5);  // 100 microseconds
      } while (ServiceManagementIsJobLoaded(label));
    }

    return true;
  }

  if (err != 0) {
    return false;
  }

  return true;
}

bool ServiceManagementIsJobLoaded(const std::string& label) {
  base::mac::ScopedLaunchData dictionary(LaunchDataDictionaryForJob(label));
  if (!dictionary.is_valid()) {
    return false;
  }

  return true;
}

pid_t ServiceManagementIsJobRunning(const std::string& label) {
  base::mac::ScopedLaunchData dictionary(LaunchDataDictionaryForJob(label));
  if (!dictionary.is_valid()) {
    return 0;
  }

  launch_data_t pid = LaunchDataDictLookup(dictionary.get(), LAUNCH_JOBKEY_PID);
  if (!pid) {
    return 0;
  }

  if (LaunchDataGetType(pid) != LAUNCH_DATA_INTEGER) {
    return 0;
  }

  return LaunchDataGetInteger(pid);
}

}  // namespace crashpad