2019-05-30 00:58:32 +02:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2019 The Android Open Source Project
|
|
|
|
*
|
|
|
|
* 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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#pragma once
|
|
|
|
|
|
|
|
#include <sys/resource.h>
|
|
|
|
#include <sys/types.h>
|
|
|
|
|
2020-06-09 06:44:17 +02:00
|
|
|
#include <optional>
|
2019-05-30 00:58:32 +02:00
|
|
|
#include <string>
|
|
|
|
#include <vector>
|
|
|
|
|
2019-09-10 23:20:35 +02:00
|
|
|
#include <android-base/unique_fd.h>
|
2019-05-30 00:58:32 +02:00
|
|
|
#include <cutils/iosched_policy.h>
|
|
|
|
|
2022-11-15 01:45:47 +01:00
|
|
|
#include "interprocess_fifo.h"
|
2020-06-09 06:44:17 +02:00
|
|
|
#include "mount_namespace.h"
|
2019-05-30 00:58:32 +02:00
|
|
|
#include "result.h"
|
|
|
|
|
|
|
|
namespace android {
|
|
|
|
namespace init {
|
|
|
|
|
2022-11-18 18:12:52 +01:00
|
|
|
// Constants used by Service::Start() for communication between parent and child.
|
|
|
|
enum ServiceCode : uint8_t {
|
|
|
|
kActivatingCgroupsFailed,
|
|
|
|
kCgroupsActivated,
|
2022-11-15 01:45:47 +01:00
|
|
|
kSetSidFinished,
|
2022-11-18 18:12:52 +01:00
|
|
|
};
|
|
|
|
|
2019-09-10 23:20:35 +02:00
|
|
|
class Descriptor {
|
|
|
|
public:
|
|
|
|
Descriptor(const std::string& name, android::base::unique_fd fd)
|
|
|
|
: name_(name), fd_(std::move(fd)){};
|
|
|
|
|
2020-11-24 20:34:40 +01:00
|
|
|
// Publish() unsets FD_CLOEXEC from the FD and publishes its name via setenv(). It should be
|
|
|
|
// called when starting a service after fork() and before exec().
|
2019-09-10 23:20:35 +02:00
|
|
|
void Publish() const;
|
|
|
|
|
|
|
|
private:
|
|
|
|
std::string name_;
|
|
|
|
android::base::unique_fd fd_;
|
|
|
|
};
|
|
|
|
|
2019-07-09 22:33:36 +02:00
|
|
|
struct SocketDescriptor {
|
|
|
|
std::string name;
|
|
|
|
int type = 0;
|
|
|
|
uid_t uid = 0;
|
|
|
|
gid_t gid = 0;
|
|
|
|
int perm = 0;
|
|
|
|
std::string context;
|
|
|
|
bool passcred = false;
|
init: Add option to listen on sockets before starting service.
Review note: Original change was a p-o-c by agl in
https://r.android.com/2094350 which I think is actually
production quality. I'm just taking it over so that he doesn't
get spammed by any review comments as that's not a good use
of his time.
Needed for the hardware entropy daemon (see bug).
Original commit message:
If one needs to create a service that synchronously starts listening on
a socket then there are currently no good options.
The traditional UNIX solution is to have the service create the socket
and then daemonise. In this situation, init could start the service with
`exec_start` and yet not block forever because the service forks and
exits. However, when the initial child process exits, init kills the
daemon process:
> init: Killed 1 additional processes from a oneshot process group for
> service 'foo'. This is new behavior, previously child processes
> would not be killed in this case.
Next, there is a `socket` option for services and (although the
documentation didn't nail this down), the socket is created
synchronously by `start`. However, init doesn't call `listen` on the
socket so, until the service starts listening on the socket itself,
clients will get ECONNREFUSED.
This this change adds a `+listen` option, similar to `+passcred` which
allows a socket service to reliably handle connections.
Bug: 243933553
Test: Started prng_seeder from init using the new listen flag
Change-Id: I91b3b2b1fd38cc3d96e19e92b76c8e95788191d5
2022-05-12 00:32:47 +02:00
|
|
|
bool listen = false;
|
2021-07-22 06:53:28 +02:00
|
|
|
bool persist = false;
|
2019-07-09 22:33:36 +02:00
|
|
|
|
2020-11-24 20:34:40 +01:00
|
|
|
// Create() creates the named unix domain socket in /dev/socket and returns a Descriptor object.
|
|
|
|
// It should be called when starting a service, before calling fork(), such that the socket is
|
|
|
|
// synchronously created before starting any other services, which may depend on it.
|
2019-09-10 23:20:35 +02:00
|
|
|
Result<Descriptor> Create(const std::string& global_context) const;
|
2019-07-09 22:33:36 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
struct FileDescriptor {
|
|
|
|
std::string name;
|
|
|
|
std::string type;
|
|
|
|
|
2019-09-10 23:20:35 +02:00
|
|
|
Result<Descriptor> Create() const;
|
2019-07-09 22:33:36 +02:00
|
|
|
};
|
|
|
|
|
2019-05-30 00:58:32 +02:00
|
|
|
struct NamespaceInfo {
|
2019-07-09 00:09:36 +02:00
|
|
|
int flags;
|
2019-05-30 00:58:32 +02:00
|
|
|
// Pair of namespace type, path to name.
|
|
|
|
std::vector<std::pair<int, std::string>> namespaces_to_enter;
|
|
|
|
};
|
2020-06-09 06:44:17 +02:00
|
|
|
Result<void> EnterNamespaces(const NamespaceInfo& info, const std::string& name,
|
|
|
|
std::optional<MountNamespace> override_mount_namespace);
|
2019-05-30 00:58:32 +02:00
|
|
|
|
|
|
|
struct ProcessAttributes {
|
|
|
|
std::string console;
|
|
|
|
IoSchedClass ioprio_class;
|
|
|
|
int ioprio_pri;
|
|
|
|
std::vector<std::pair<int, rlimit>> rlimits;
|
2023-04-04 01:29:22 +02:00
|
|
|
std::optional<uid_t> parsed_uid;
|
2019-05-30 00:58:32 +02:00
|
|
|
gid_t gid;
|
|
|
|
std::vector<gid_t> supp_gids;
|
|
|
|
int priority;
|
2019-09-24 01:16:54 +02:00
|
|
|
bool stdio_to_kmsg;
|
2023-04-04 01:29:22 +02:00
|
|
|
|
|
|
|
uid_t uid() const { return parsed_uid.value_or(0); }
|
2019-05-30 00:58:32 +02:00
|
|
|
};
|
2022-11-15 01:54:03 +01:00
|
|
|
|
|
|
|
inline bool RequiresConsole(const ProcessAttributes& attr) {
|
|
|
|
return !attr.console.empty();
|
|
|
|
}
|
|
|
|
|
2022-11-15 01:45:47 +01:00
|
|
|
Result<void> SetProcessAttributes(const ProcessAttributes& attr, InterprocessFifo setsid_finished);
|
2019-05-30 00:58:32 +02:00
|
|
|
|
2019-06-10 20:08:01 +02:00
|
|
|
Result<void> WritePidToFiles(std::vector<std::string>* files);
|
2019-05-30 00:58:32 +02:00
|
|
|
|
|
|
|
} // namespace init
|
|
|
|
} // namespace android
|