/* Copyright (C) 2017 Free Software Foundation, Inc. This file is part of the GNU Hurd. The GNU Hurd is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 2, or (at your option) any later version. The GNU Hurd is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with the GNU Hurd. If not, see . */ /* Translator initialization and demuxing */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "libnetfs/io_S.h" #include "libnetfs/fs_S.h" #include "libports/notify_S.h" #include "libnetfs/fsys_S.h" #include "libports/interrupt_S.h" #include "libnetfs/ifsock_S.h" #include "libmachdev/machdev.h" #include "libshouldbeinlibc/wire.h" #include #include #include "pcifs.h" struct pcifs *fs; volatile struct mapped_time_value *pcifs_maptime; /* Libnetfs stuff */ int netfs_maxsymlinks = 0; char *netfs_server_name = "pci-arbiter"; char *netfs_server_version = HURD_VERSION; static mach_port_t pci_control_port; static io_return_t pci_device_open (mach_port_t reply_port, mach_msg_type_name_t reply_port_type, dev_mode_t mode, const char *name, device_t * devp, mach_msg_type_name_t * devicePoly) { io_return_t err = D_SUCCESS; mach_port_t dev_master, root; string_t retry_name; retry_type retry; uid_t idlist[] = {0, 0, 0}; if (strncmp(name, "pci", 3)) err = D_NO_SUCH_DEVICE; /* Fall back to opening kernel device master */ if (err) { err = get_privileged_ports(NULL, &dev_master); if (err) return err; if (dev_master == MACH_PORT_NULL) return D_NO_SUCH_DEVICE; err = device_open (dev_master, mode, name, devp); mach_port_deallocate (mach_task_self (), dev_master); if (err) return err; *devicePoly = MACH_MSG_TYPE_MOVE_SEND; return D_SUCCESS; } err = fsys_getroot(pci_control_port, MACH_PORT_NULL, MACH_MSG_TYPE_COPY_SEND, idlist, 3, idlist, 3, 0, &retry, retry_name, &root); if (err) return err; *devp = root; *devicePoly = MACH_MSG_TYPE_COPY_SEND; return D_SUCCESS; } static struct machdev_device_emulation_ops pci_arbiter_emulation_ops = { NULL, NULL, NULL, NULL, pci_device_open, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, }; int netfs_demuxer (mach_msg_header_t * inp, mach_msg_header_t * outp) { mig_routine_t routine; if ((routine = netfs_io_server_routine (inp)) || (routine = netfs_fs_server_routine (inp)) || (routine = ports_notify_server_routine (inp)) || (routine = netfs_fsys_server_routine (inp)) || (routine = ports_interrupt_server_routine (inp)) || (routine = netfs_ifsock_server_routine (inp)) || (routine = pci_server_routine (inp)) || (routine = startup_notify_server_routine (inp))) { (*routine) (inp, outp); return TRUE; } else return FALSE; } static void * netfs_server_func (void *arg) { error_t err; do { ports_manage_port_operations_multithread (netfs_port_bucket, netfs_demuxer, 1000 * 60 * 2, /* two minutes thread */ 1000 * 60 * 10,/* ten minutes server */ 0); err = netfs_shutdown (0); } while (err); return NULL; } static mach_port_t pcifs_startup(mach_port_t bootstrap, int flags) { error_t err; mach_port_t realnode; struct port_info *newpi; err = ports_create_port (netfs_control_class, netfs_port_bucket, sizeof (struct port_info), &newpi); if (err) error (11, err, "Translator startup failure: pcifs_startup"); pci_control_port = ports_get_send_right (newpi); if (bootstrap != MACH_PORT_NULL) { err = fsys_startup (bootstrap, flags, pci_control_port, MACH_MSG_TYPE_COPY_SEND, &realnode); assert_perror_backtrace (err); } return realnode; } int main (int argc, char **argv) { error_t err; mach_port_t bootstrap; mach_port_t next_task; pthread_t t, mt; file_t underlying_node = MACH_PORT_NULL; /* Parse options */ alloc_file_system (&fs); argp_parse (netfs_runtime_argp, argc, argv, 0, 0, 0); next_task = fs->params.next_task; if (next_task != MACH_PORT_NULL) { /* We are a bootstrap process */ machdev_register (&pci_arbiter_emulation_ops); /* TODO: make libmachdev allow us to also run netfs on the translated path, * so that we don't need a second pci-arbiter to serve /servers/bus/pci. */ machdev_trivfs_init (argc, argv, next_task, "pci", NULL /* _SERVERS_BUS "/pci" */, &bootstrap); /* Make sure we will not swap out, in case we are needed for swapping back in. */ err = wire_task_self (); if (err) error (1, errno, "cannot lock all memory"); machdev_device_init (); err = pthread_create (&t, NULL, machdev_server, NULL); if (err) error (1, err, "Creating machdev_server thread"); pthread_detach (t); } else { task_get_bootstrap_port (mach_task_self (), &bootstrap); if (bootstrap == MACH_PORT_NULL) error (1, 0, "must be started as a translator"); } /* Initialize netfs and start the translator. */ netfs_init (); err = maptime_map (0, 0, &pcifs_maptime); if (err) err = maptime_map (1, 0, &pcifs_maptime); if (err) error (1, err, "mapping time"); /* Start the PCI system: NB: pciaccess will choose x86 first and take lock */ err = pci_system_init (); if (err) error (1, err, "Starting the PCI system"); if (next_task != MACH_PORT_NULL) machdev_trivfs_server_startup (bootstrap); if (next_task == MACH_PORT_NULL) underlying_node = netfs_startup (bootstrap, O_READ); /* Create the root node first */ err = init_root_node (underlying_node); if (err) error (1, err, "Creating the root node"); if (next_task != MACH_PORT_NULL) pcifs_startup (bootstrap, O_READ); err = init_file_system (fs); if (err) error (1, err, "Creating the PCI filesystem"); /* Create the filesystem tree */ err = create_fs_tree (fs); if (err) error (1, err, "Creating the PCI filesystem tree"); /* Set permissions */ err = fs_set_permissions (fs); if (err) error (1, err, "Setting permissions"); if (next_task != MACH_PORT_NULL) { err = pthread_create(&mt, NULL, machdev_trivfs_server_loop, NULL); if (err) error(1, err, "Creating machdev_trivfs_server_loop thread"); pthread_detach(mt); } netfs_server_func (NULL); /* Never reached */ return 0; }