rpadlpar_core.c
来自「优龙2410linux2.6.8内核源代码」· C语言 代码 · 共 398 行
C
398 行
/* * Interface for Dynamic Logical Partitioning of I/O Slots on * RPA-compliant PPC64 platform. * * John Rose <johnrose@austin.ibm.com> * Linda Xie <lxie@us.ibm.com> * * October 2003 * * Copyright (C) 2003 IBM. * * This program 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 of the License, or (at your option) any later version. */#include <linux/init.h>#include <linux/pci.h>#include <asm/pci-bridge.h>#include <asm/semaphore.h>#include <asm/rtas.h>#include "../pci.h"#include "rpaphp.h"#include "rpadlpar.h"static DECLARE_MUTEX(rpadlpar_sem);static struct device_node *find_php_slot_vio_node(char *drc_name){ struct device_node *child; struct device_node *parent = of_find_node_by_name(NULL, "vdevice"); char *loc_code; if (!parent) return NULL; for (child = of_get_next_child(parent, NULL); child; child = of_get_next_child(parent, child)) { loc_code = get_property(child, "ibm,loc-code", NULL); if (loc_code && !strncmp(loc_code, drc_name, strlen(drc_name))) return child; } return NULL;}static struct device_node *find_php_slot_pci_node(char *drc_name){ struct device_node *np = NULL; char *name; while ((np = of_find_node_by_type(np, "pci"))) if (is_hotplug_capable(np)) { name = rpaphp_get_drc_name(np); if (name && (!strcmp(drc_name, name))) break; } return np;}static struct slot *find_slot(char *drc_name){ struct list_head *tmp, *n; struct slot *slot; list_for_each_safe(tmp, n, &rpaphp_slot_head) { slot = list_entry(tmp, struct slot, rpaphp_slot_list); if (strcmp(slot->location, drc_name) == 0) return slot; } return NULL;}static void rpadlpar_claim_one_bus(struct pci_bus *b){ struct list_head *ld; struct pci_bus *child_bus; for (ld = b->devices.next; ld != &b->devices; ld = ld->next) { struct pci_dev *dev = pci_dev_b(ld); int i; for (i = 0; i < PCI_NUM_RESOURCES; i++) { struct resource *r = &dev->resource[i]; if (r->parent || !r->start || !r->flags) continue; rpaphp_claim_resource(dev, i); } } list_for_each_entry(child_bus, &b->children, node) rpadlpar_claim_one_bus(child_bus);}static int pci_add_secondary_bus(struct device_node *dn, struct pci_dev *bridge_dev){ struct pci_controller *hose = dn->phb; struct pci_bus *child; u8 sec_busno; /* Get busno of downstream bus */ pci_read_config_byte(bridge_dev, PCI_SECONDARY_BUS, &sec_busno); /* Allocate and add to children of bridge_dev->bus */ child = pci_add_new_bus(bridge_dev->bus, bridge_dev, sec_busno); if (!child) { printk(KERN_ERR "%s: could not add secondary bus\n", __FUNCTION__); return 1; } sprintf(child->name, "PCI Bus #%02x", child->number); /* Fixup subordinate bridge bases and resources */ pcibios_fixup_bus(child); /* Claim new bus resources */ rpadlpar_claim_one_bus(bridge_dev->bus); if (hose->last_busno < child->number) hose->last_busno = child->number; dn->bussubno = child->number; /* ioremap() for child bus */ if (remap_bus_range(child)) { printk(KERN_ERR "%s: could not ioremap() child bus\n", __FUNCTION__); return 1; } return 0;}static struct pci_dev *dlpar_pci_add_bus(struct device_node *dn){ struct pci_controller *hose = dn->phb; struct pci_dev *dev = NULL; /* Scan phb bus for EADS device, adding new one to bus->devices */ if (!pci_scan_single_device(hose->bus, dn->devfn)) { printk(KERN_ERR "%s: found no device on bus\n", __FUNCTION__); return NULL; } /* Add new devices to global lists. Register in proc, sysfs. */ pci_bus_add_devices(hose->bus); /* Confirm new bridge dev was created */ dev = rpaphp_find_pci_dev(dn); if (!dev) { printk(KERN_ERR "%s: failed to add pci device\n", __FUNCTION__); return NULL; } if (dev->hdr_type != PCI_HEADER_TYPE_BRIDGE) { printk(KERN_ERR "%s: unexpected header type %d\n", __FUNCTION__, dev->hdr_type); return NULL; } if (pci_add_secondary_bus(dn, dev)) return NULL; return dev;}static int dlpar_pci_remove_bus(struct pci_dev *bridge_dev){ struct pci_bus *secondary_bus; if (!bridge_dev) { printk(KERN_ERR "%s: unexpected null device\n", __FUNCTION__); return 1; } secondary_bus = bridge_dev->subordinate; if (unmap_bus_range(secondary_bus)) { printk(KERN_ERR "%s: failed to unmap bus range\n", __FUNCTION__); return 1; } pci_remove_bus_device(bridge_dev); return 0;}static inline int dlpar_add_pci_slot(char *drc_name, struct device_node *dn){ struct pci_dev *dev; /* Add pci bus */ dev = dlpar_pci_add_bus(dn); if (!dev) { printk(KERN_ERR "%s: unable to add bus %s\n", __FUNCTION__, drc_name); return -EIO; } return 0;}/** * dlpar_add_slot - DLPAR add an I/O Slot * @drc_name: drc-name of newly added slot * * Make the hotplug module and the kernel aware * of a newly added I/O Slot. * Return Codes - * 0 Success * -ENODEV Not a valid drc_name * -EINVAL Slot already added * -ERESTARTSYS Signalled before obtaining lock * -EIO Internal PCI Error */int dlpar_add_slot(char *drc_name){ struct device_node *dn; int rc = 0; if (down_interruptible(&rpadlpar_sem)) return -ERESTARTSYS; /* Check for existing hotplug slot */ if (find_slot(drc_name)) { rc = -EINVAL; goto exit; } dn = find_php_slot_vio_node(drc_name); if (!dn) { dn = find_php_slot_pci_node(drc_name); if (dn) rc = dlpar_add_pci_slot(drc_name, dn); else { rc = -ENODEV; goto exit; } } /* Add hotplug slot for new VIOA or PCI */ if (!rc && rpaphp_add_slot(dn)) { printk(KERN_ERR "%s: unable to add hotplug slot %s\n", __FUNCTION__, drc_name); rc = -EIO; }exit: up(&rpadlpar_sem); return rc;}/** * dlpar_remove_vio_slot - DLPAR remove a virtual I/O Slot * @drc_name: drc-name of newly added slot * * Remove the kernel and hotplug representations * of an I/O Slot. * Return Codes: * 0 Success * -EIO Internal Error */int dlpar_remove_vio_slot(struct slot *slot, char *drc_name){ /* Remove hotplug slot */ if (rpaphp_remove_slot(slot)) { printk(KERN_ERR "%s: unable to remove hotplug slot %s\n", __FUNCTION__, drc_name); return -EIO; } return 0;}/** * dlpar_remove_slot - DLPAR remove a PCI I/O Slot * @drc_name: drc-name of newly added slot * * Remove the kernel and hotplug representations * of a PCI I/O Slot. * Return Codes: * 0 Success * -ENODEV Not a valid drc_name * -EIO Internal PCI Error */int dlpar_remove_pci_slot(struct slot *slot, char *drc_name){ struct pci_dev *bridge_dev; bridge_dev = slot->bridge; if (!bridge_dev) { printk(KERN_ERR "%s: unexpected null bridge device\n", __FUNCTION__); return -EIO; } /* Remove hotplug slot */ if (rpaphp_remove_slot(slot)) { printk(KERN_ERR "%s: unable to remove hotplug slot %s\n", __FUNCTION__, drc_name); return -EIO; } /* Remove pci bus */ if (dlpar_pci_remove_bus(bridge_dev)) { printk(KERN_ERR "%s: unable to remove pci bus %s\n", __FUNCTION__, drc_name); return -EIO; } return 0;}/** * dlpar_remove_slot - DLPAR remove an I/O Slot * @drc_name: drc-name of newly added slot * * Remove the kernel and hotplug representations * of an I/O Slot. * Return Codes: * 0 Success * -ENODEV Not a valid drc_name * -EINVAL Slot already removed * -ERESTARTSYS Signalled before obtaining lock * -EIO Internal Error */int dlpar_remove_slot(char *drc_name){ struct slot *slot; int rc = 0; if (down_interruptible(&rpadlpar_sem)) return -ERESTARTSYS; if (!find_php_slot_vio_node(drc_name) && !find_php_slot_pci_node(drc_name)) { rc = -ENODEV; goto exit; } slot = find_slot(drc_name); if (!slot) { rc = -EINVAL; goto exit; } switch (slot->dev_type) { case PCI_DEV: rc = dlpar_remove_pci_slot(slot, drc_name); break; case VIO_DEV: rc = dlpar_remove_vio_slot(slot, drc_name); break; default: rc = -EIO; }exit: up(&rpadlpar_sem); return rc;}static inline int is_dlpar_capable(void){ int rc = rtas_token("ibm,configure-connector"); return (int) (rc != RTAS_UNKNOWN_SERVICE);}int __init rpadlpar_io_init(void){ int rc = 0; if (!is_dlpar_capable()) { printk(KERN_WARNING "%s: partition not DLPAR capable\n", __FUNCTION__); return -EPERM; } rc = dlpar_sysfs_init(); return rc;}void rpadlpar_io_exit(void){ dlpar_sysfs_exit(); return;}module_init(rpadlpar_io_init);module_exit(rpadlpar_io_exit);MODULE_LICENSE("GPL");
⌨️ 快捷键说明
复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?