blob: cc6524cc106fd91857e3f2ee96ba0a786de6632e [file] [log] [blame]
// Copyright 2018 The Fuchsia Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include <fbl/algorithm.h>
#include <lib/async-loop/cpp/loop.h>
#include <lib/crashanalyzer/crashanalyzer.h>
#include <lib/fdio/util.h>
#include <lib/process-launcher/launcher.h>
#include <lib/svc/outgoing.h>
#include <lib/sysmem/sysmem.h>
#include <zircon/process.h>
#include <zircon/processargs.h>
#include <zircon/status.h>
// An instance of a zx_service_provider_t.
//
// Includes the |ctx| pointer for the zx_service_provider_t.
typedef struct zx_service_provider_instance {
// The service provider for which this structure is an instance.
const zx_service_provider_t* provider;
// The |ctx| pointer returned by the provider's |init| function, if any.
void* ctx;
} zx_service_provider_instance_t;
static zx_status_t provider_init(zx_service_provider_instance_t* instance) {
if (instance->provider->ops->init) {
zx_status_t status = instance->provider->ops->init(&instance->ctx);
if (status != ZX_OK)
return status;
}
return ZX_OK;
}
static zx_status_t provider_publish(zx_service_provider_instance_t* instance,
async_dispatcher_t* dispatcher, const fbl::RefPtr<fs::PseudoDir>& dir) {
const zx_service_provider_t* provider = instance->provider;
if (!provider->services || !provider->ops->connect)
return ZX_ERR_INVALID_ARGS;
for (size_t i = 0; provider->services[i]; ++i) {
const char* service_name = provider->services[i];
zx_status_t status = dir->AddEntry(
service_name,
fbl::MakeRefCounted<fs::Service>([instance, dispatcher, service_name](zx::channel request) {
return instance->provider->ops->connect(instance->ctx, dispatcher, service_name, request.release());
}));
if (status != ZX_OK) {
for (size_t j = 0; j < i; ++j)
dir->RemoveEntry(provider->services[j]);
return status;
}
}
return ZX_OK;
}
static void provider_release(zx_service_provider_instance_t* instance) {
if (instance->provider->ops->release)
instance->provider->ops->release(instance->ctx);
instance->ctx = nullptr;
}
static zx_status_t provider_load(zx_service_provider_instance_t* instance,
async_dispatcher_t* dispatcher, const fbl::RefPtr<fs::PseudoDir>& dir) {
if (instance->provider->version != SERVICE_PROVIDER_VERSION) {
return ZX_ERR_INVALID_ARGS;
}
zx_status_t status = provider_init(instance);
if (status != ZX_OK) {
return status;
}
status = provider_publish(instance, dispatcher, dir);
if (status != ZX_OK) {
provider_release(instance);
return status;
}
return ZX_OK;
}
static zx_handle_t appmgr_svc;
// We should host the tracelink service ourselves instead of routing the request
// to appmgr.
zx_status_t publish_tracelink(const fbl::RefPtr<fs::PseudoDir>& dir) {
const char* service_name = "fuchsia.tracelink.Registry";
return dir->AddEntry(
service_name,
fbl::MakeRefCounted<fs::Service>([service_name](zx::channel request) {
return fdio_service_connect_at(appmgr_svc, service_name, request.release());
}));
}
// We shouldn't need to access these non-Zircon services from svchost, but
// currently some tests assume they can reach these services from the test
// environment. Instead, we should make the test environment hermetic and
// remove the dependencies on these services.
static constexpr const char* deprecated_services[] = {
// remove amber.Control when CP-50 is resolved
"fuchsia.amber.Control",
"fuchsia.cobalt.CobaltEncoderFactory",
"fuchsia.devicesettings.DeviceSettingsManager",
"fuchsia.logger.Log",
"fuchsia.logger.LogSink",
"fuchsia.media.Audio",
"fuchsia.mediaplayer.MediaPlayer",
"fuchsia.net.LegacySocketProvider",
// Legacy interface for netstack, defined in //garnet
"fuchsia.netstack.Netstack",
// New interface for netstack (WIP), defined in //zircon
"fuchsia.net_stack.Stack",
"fuchsia.power.PowerManager",
"fuchsia.sys.Environment",
"fuchsia.sys.Launcher",
"fuchsia.wlan.service.Wlan",
// fdio name for Netstack. Will be removed with the new interfaces defined
// in NET-863.
"net.Netstack",
// TODO(IN-458): This entry is temporary, until IN-458 is resolved.
"fuchsia.tracing.TraceController",
nullptr,
// DO NOT ADD MORE ENTRIES TO THIS LIST.
// Tests should not be accessing services from the environment. Instead,
// they should run in containers that have their own service instances.
};
void publish_deprecated_services(const fbl::RefPtr<fs::PseudoDir>& dir) {
for (size_t i = 0; deprecated_services[i]; ++i) {
const char* service_name = deprecated_services[i];
dir->AddEntry(
service_name,
fbl::MakeRefCounted<fs::Service>([service_name](zx::channel request) {
return fdio_service_connect_at(appmgr_svc, service_name, request.release());
}));
}
}
int main(int argc, char** argv) {
async::Loop loop(&kAsyncLoopConfigNoAttachToThread);
async_dispatcher_t* dispatcher = loop.dispatcher();
svc::Outgoing outgoing(dispatcher);
appmgr_svc = zx_take_startup_handle(PA_HND(PA_USER0, 0));
zx_status_t status = outgoing.ServeFromStartupInfo();
if (status != ZX_OK) {
fprintf(stderr, "svchost: error: Failed to serve outgoing directory: %d (%s).\n",
status, zx_status_get_string(status));
return 1;
}
zx_service_provider_instance_t service_providers[] = {
{.provider = crashanalyzer_get_service_provider(), .ctx = nullptr},
{.provider = launcher_get_service_provider(), .ctx = nullptr},
{.provider = sysmem_get_service_provider(), .ctx = nullptr},
};
for (size_t i = 0; i < fbl::count_of(service_providers); ++i) {
status = provider_load(&service_providers[i], dispatcher, outgoing.public_dir());
if (status != ZX_OK) {
fprintf(stderr, "svchost: error: Failed to load service provider %zu: %d (%s).\n",
i, status, zx_status_get_string(status));
return 1;
}
}
status = publish_tracelink(outgoing.public_dir());
if (status != ZX_OK) {
fprintf(stderr, "svchost: error: Failed to publish tracelink: %d (%s).\n",
status, zx_status_get_string(status));
return 1;
}
publish_deprecated_services(outgoing.public_dir());
status = loop.Run();
for (size_t i = 0; i < fbl::count_of(service_providers); ++i) {
provider_release(&service_providers[i]);
}
return status;
}