⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 shpchp_ctrl.c

📁 h内核
💻 C
📖 第 1 页 / 共 5 页
字号:
	if (next == NULL) {		return(1);	}	if (next == bridge) {		shpchp_slot_list[bridge->bus] = bridge->next;		kfree(bridge);		return(0);	}	while ((next->next != bridge) && (next->next != NULL)) {		next = next->next;	}	if (next->next == bridge) {		next->next = bridge->next;		kfree(bridge);		return(0);	} else		return(2);}/** * shpchp_slot_find - Looks for a node by bus, and device, multiple functions accessed * @bus: bus to find * @device: device to find * @index: is 0 for first function found, 1 for the second... * * Returns pointer to the node if successful, %NULL otherwise. */struct pci_func *shpchp_slot_find(u8 bus, u8 device, u8 index){	int found = -1;	struct pci_func *func;	func = shpchp_slot_list[bus];	if ((func == NULL) || ((func->device == device) && (index == 0)))		return(func);	if (func->device == device)		found++;	while (func->next != NULL) {		func = func->next;		if (func->device == device)			found++;		if (found == index)			return(func);	}	return(NULL);}static int is_bridge(struct pci_func * func){	/* Check the header type */	if (((func->config_space[0x03] >> 16) & 0xFF) == 0x01)		return 1;	else		return 0;}/* The following routines constitute the bulk of the    hotplug controller logic */static u32 change_bus_speed(struct controller *ctrl, struct slot *p_slot, enum pci_bus_speed speed){ 	u32 rc = 0;	dbg("%s: change to speed %d\n", __FUNCTION__, speed);	down(&ctrl->crit_sect);	if ((rc = p_slot->hpc_ops->set_bus_speed_mode(p_slot, speed))) {		err("%s: Issue of set bus speed mode command failed\n", __FUNCTION__);		up(&ctrl->crit_sect);		return WRONG_BUS_FREQUENCY;	}	wait_for_ctrl_irq (ctrl);			if ((rc = p_slot->hpc_ops->check_cmd_status(ctrl))) {		err("%s: Can't set bus speed/mode in the case of adapter & bus mismatch\n",			  __FUNCTION__);		err("%s: Error code (%d)\n", __FUNCTION__, rc);		up(&ctrl->crit_sect);		return WRONG_BUS_FREQUENCY;	}	up(&ctrl->crit_sect);	return rc;}static u32 fix_bus_speed(struct controller *ctrl, struct slot *pslot, u8 flag, enum pci_bus_speed asp, enum pci_bus_speed bsp, enum pci_bus_speed msp){ 	u32 rc = 0;		if (flag != 0) { /* Other slots on the same bus are occupied */		if ( asp < bsp ) {			err("%s: speed of bus %x and adapter %x mismatch\n", __FUNCTION__, bsp, asp);			return WRONG_BUS_FREQUENCY;		}	} else {		/* Other slots on the same bus are empty */		if (msp == bsp) {		/* if adapter_speed >= bus_speed, do nothing */			if (asp < bsp) {				/* 				* Try to lower bus speed to accommodate the adapter if other slots 				* on the same controller are empty				*/				if ((rc = change_bus_speed(ctrl, pslot, asp)))					return rc;			} 		} else {			if (asp < msp) {				if ((rc = change_bus_speed(ctrl, pslot, asp)))					return rc;			} else {				if ((rc = change_bus_speed(ctrl, pslot, msp)))					return rc;			}		}	}	return rc;}/** * board_added - Called after a board has been added to the system. * * Turns power on for the board * Configures board * */static u32 board_added(struct pci_func * func, struct controller * ctrl){	u8 hp_slot;	u8 slots_not_empty = 0;	int index;	u32 temp_register = 0xFFFFFFFF;	u32 retval, rc = 0;	struct pci_func *new_func = NULL;	struct slot *p_slot;	struct resource_lists res_lists;	enum pci_bus_speed adapter_speed, bus_speed, max_bus_speed;	u8 pi, mode;	p_slot = shpchp_find_slot(ctrl, func->device);	hp_slot = func->device - ctrl->slot_device_offset;	dbg("%s: func->device, slot_offset, hp_slot = %d, %d ,%d\n", __FUNCTION__, func->device, ctrl->slot_device_offset, hp_slot);	/* Wait for exclusive access to hardware */	down(&ctrl->crit_sect);	/* Power on slot without connecting to bus */	rc = p_slot->hpc_ops->power_on_slot(p_slot);	if (rc) {		err("%s: Failed to power on slot\n", __FUNCTION__);		/* Done with exclusive hardware access */		up(&ctrl->crit_sect);		return -1;	}				/* Wait for the command to complete */	wait_for_ctrl_irq (ctrl);		rc = p_slot->hpc_ops->check_cmd_status(ctrl);	if (rc) {		err("%s: Failed to power on slot, error code(%d)\n", __FUNCTION__, rc);		/* Done with exclusive hardware access */		up(&ctrl->crit_sect);		return -1;	}		if ((ctrl->pci_dev->vendor == 0x8086) && (ctrl->pci_dev->device == 0x0332)) {		if (slots_not_empty)			return WRONG_BUS_FREQUENCY;				if ((rc = p_slot->hpc_ops->set_bus_speed_mode(p_slot, PCI_SPEED_33MHz))) {			err("%s: Issue of set bus speed mode command failed\n", __FUNCTION__);			up(&ctrl->crit_sect);			return WRONG_BUS_FREQUENCY;		}		wait_for_ctrl_irq (ctrl);				if ((rc = p_slot->hpc_ops->check_cmd_status(ctrl))) {			err("%s: Can't set bus speed/mode in the case of adapter & bus mismatch\n",				  __FUNCTION__);			err("%s: Error code (%d)\n", __FUNCTION__, rc);			up(&ctrl->crit_sect);			return WRONG_BUS_FREQUENCY;		}		/* turn on board, blink green LED, turn off Amber LED */		if ((rc = p_slot->hpc_ops->slot_enable(p_slot))) {			err("%s: Issue of Slot Enable command failed\n", __FUNCTION__);			up(&ctrl->crit_sect);			return rc;		}		wait_for_ctrl_irq (ctrl);		if ((rc = p_slot->hpc_ops->check_cmd_status(ctrl))) {			err("%s: Failed to enable slot, error code(%d)\n", __FUNCTION__, rc);			up(&ctrl->crit_sect);			return rc;  		}	} 	rc = p_slot->hpc_ops->get_adapter_speed(p_slot, &adapter_speed);	/* 0 = PCI 33Mhz, 1 = PCI 66 Mhz, 2 = PCI-X 66 PA, 4 = PCI-X 66 ECC, */	/* 5 = PCI-X 133 PA, 7 = PCI-X 133 ECC,  0xa = PCI-X 133 Mhz 266, */	/* 0xd = PCI-X 133 Mhz 533 */	/* This encoding is different from the one used in cur_bus_speed & */	/* max_bus_speed */	if (rc  || adapter_speed == PCI_SPEED_UNKNOWN) {		err("%s: Can't get adapter speed or bus mode mismatch\n", __FUNCTION__);		/* Done with exclusive hardware access */		up(&ctrl->crit_sect);		return WRONG_BUS_FREQUENCY;	}	rc = p_slot->hpc_ops->get_cur_bus_speed(p_slot, &bus_speed);	if (rc || bus_speed == PCI_SPEED_UNKNOWN) {		err("%s: Can't get bus operation speed\n", __FUNCTION__);		/* Done with exclusive hardware access */		up(&ctrl->crit_sect);		return WRONG_BUS_FREQUENCY;	}	rc = p_slot->hpc_ops->get_max_bus_speed(p_slot, &max_bus_speed);	if (rc || max_bus_speed == PCI_SPEED_UNKNOWN) {		err("%s: Can't get max bus operation speed\n", __FUNCTION__);		max_bus_speed = bus_speed;	}	/* Done with exclusive hardware access */	up(&ctrl->crit_sect);	if ((rc  = p_slot->hpc_ops->get_prog_int(p_slot, &pi))) {		err("%s: Can't get controller programming interface, set it to 1\n", __FUNCTION__);		pi = 1;	}	/* Check if there are other slots or devices on the same bus */	if (!list_empty(&ctrl->pci_dev->subordinate->devices))		slots_not_empty = 1;	dbg("%s: slots_not_empty %d, pi %d\n", __FUNCTION__, 		slots_not_empty, pi);	dbg("adapter_speed %d, bus_speed %d, max_bus_speed %d\n", 		adapter_speed, bus_speed, max_bus_speed);	if (pi == 2) {		dbg("%s: In PI = %d\n", __FUNCTION__, pi);		if ((rc = p_slot->hpc_ops->get_mode1_ECC_cap(p_slot, &mode))) {			err("%s: Can't get Mode1_ECC, set mode to 0\n", __FUNCTION__);			mode = 0;		}		switch (adapter_speed) {		case PCI_SPEED_133MHz_PCIX_533:		case PCI_SPEED_133MHz_PCIX_266:			if ((bus_speed != adapter_speed) &&			   ((rc = fix_bus_speed(ctrl, p_slot, slots_not_empty, adapter_speed, bus_speed, max_bus_speed)))) 				return rc;			break;			case PCI_SPEED_133MHz_PCIX_ECC:		case PCI_SPEED_133MHz_PCIX:			if (mode) { /* Bus - Mode 1 ECC */				if ((bus_speed != 0x7) &&				   ((rc = fix_bus_speed(ctrl, p_slot, slots_not_empty, adapter_speed, bus_speed, max_bus_speed)))) 					return rc;			} else {				if ((bus_speed != 0x4) &&				   ((rc = fix_bus_speed(ctrl, p_slot, slots_not_empty, adapter_speed, bus_speed, max_bus_speed)))) 					return rc;			}			break;		case PCI_SPEED_66MHz_PCIX_ECC:		case PCI_SPEED_66MHz_PCIX:			if (mode) { /* Bus - Mode 1 ECC */				if ((bus_speed != 0x5) &&				   ((rc = fix_bus_speed(ctrl, p_slot, slots_not_empty, adapter_speed, bus_speed, max_bus_speed)))) 					return rc;			} else {				if ((bus_speed != 0x2) &&				   ((rc = fix_bus_speed(ctrl, p_slot, slots_not_empty, adapter_speed, bus_speed, max_bus_speed)))) 					return rc;			}			break;		case PCI_SPEED_66MHz:			if ((bus_speed != 0x1) &&			   ((rc = fix_bus_speed(ctrl, p_slot, slots_not_empty, adapter_speed, bus_speed, max_bus_speed)))) 				return rc;			break;			case PCI_SPEED_33MHz:			if (bus_speed > 0x0) {				if (slots_not_empty == 0) {					if ((rc = change_bus_speed(ctrl, p_slot, adapter_speed)))						return rc;				} else {					err("%s: speed of bus %x and adapter %x mismatch\n", __FUNCTION__, bus_speed, adapter_speed);					return WRONG_BUS_FREQUENCY;				}			}			break;		default:			err("%s: speed of bus %x and adapter %x mismatch\n", __FUNCTION__, bus_speed, adapter_speed);			return WRONG_BUS_FREQUENCY;		}	} else {		/* If adpater_speed == bus_speed, nothing to do here */		dbg("%s: In PI = %d\n", __FUNCTION__, pi);		if ((adapter_speed != bus_speed) &&		   ((rc = fix_bus_speed(ctrl, p_slot, slots_not_empty, adapter_speed, bus_speed, max_bus_speed))))				return rc;	}	down(&ctrl->crit_sect);	/* turn on board, blink green LED, turn off Amber LED */	if ((rc = p_slot->hpc_ops->slot_enable(p_slot))) {		err("%s: Issue of Slot Enable command failed\n", __FUNCTION__);		up(&ctrl->crit_sect);		return rc;	}	wait_for_ctrl_irq (ctrl);	if ((rc = p_slot->hpc_ops->check_cmd_status(ctrl))) {		err("%s: Failed to enable slot, error code(%d)\n", __FUNCTION__, rc);		up(&ctrl->crit_sect);		return rc;  	}	up(&ctrl->crit_sect);	/* Wait for ~1 second */	dbg("%s: before long_delay\n", __FUNCTION__);	wait_for_ctrl_irq (ctrl);	dbg("%s: after long_delay\n", __FUNCTION__);	dbg("%s: func status = %x\n", __FUNCTION__, func->status);	/* Check for a power fault */	if (func->status == 0xFF) {		/* power fault occurred, but it was benign */		temp_register = 0xFFFFFFFF;		dbg("%s: temp register set to %x by power fault\n", __FUNCTION__, temp_register);		rc = POWER_FAILURE;		func->status = 0;	} else {		/* Get vendor/device ID u32 */		rc = pci_bus_read_config_dword (ctrl->pci_dev->subordinate, PCI_DEVFN(func->device, func->function), 			PCI_VENDOR_ID, &temp_register);		dbg("%s: pci_bus_read_config_dword returns %d\n", __FUNCTION__, rc);		dbg("%s: temp_register is %x\n", __FUNCTION__, temp_register);		if (rc != 0) {			/* Something's wrong here */			temp_register = 0xFFFFFFFF;			dbg("%s: temp register set to %x by error\n", __FUNCTION__, temp_register);		}		/* Preset return code.  It will be changed later if things go okay. */		rc = NO_ADAPTER_PRESENT;	}	/* All F's is an empty slot or an invalid board */	if (temp_register != 0xFFFFFFFF) {	  /* Check for a board in the slot */		res_lists.io_head = ctrl->io_head;		res_lists.mem_head = ctrl->mem_head;		res_lists.p_mem_head = ctrl->p_mem_head;		res_lists.bus_head = ctrl->bus_head;		res_lists.irqs = NULL;		rc = configure_new_device(ctrl, func, 0, &res_lists, 0, 0);		dbg("%s: back from configure_new_device\n", __FUNCTION__);		ctrl->io_head = res_lists.io_head;		ctrl->mem_head = res_lists.mem_head;		ctrl->p_mem_head = res_lists.p_mem_head;		ctrl->bus_head = res_lists.bus_head;		shpchp_resource_sort_and_combine(&(ctrl->mem_head));		shpchp_resource_sort_and_combine(&(ctrl->p_mem_head));		shpchp_resource_sort_and_combine(&(ctrl->io_head));		shpchp_resource_sort_and_combine(&(ctrl->bus_head));		if (rc) {			/* Wait for exclusive access to hardware */			down(&ctrl->crit_sect);			/* turn off slot, turn on Amber LED, turn off Green LED */			retval = p_slot->hpc_ops->slot_disable(p_slot);			if (retval) {				err("%s: Issue of Slot Enable command failed\n", __FUNCTION__);				/* Done with exclusive hardware access */				up(&ctrl->crit_sect);				return retval;			}			/* Wait for the command to complete */			wait_for_ctrl_irq (ctrl);			retval = p_slot->hpc_ops->check_cmd_status(ctrl);			if (retval) {				err("%s: Failed to disable slot, error code(%d)\n", __FUNCTION__, retval);				/* Done with exclusive hardware access */				up(&ctrl->crit_sect);				return retval;  			}			/* Done with exclusive hardware access */			up(&ctrl->crit_sect);			return(rc);		}		shpchp_save_slot_config(ctrl, func);		func->status = 0;		func->switch_save = 0x10;		func->is_a_board = 0x01;		func->pwr_save = 1;		/* Next, we will instantiate the linux pci_dev structures 		 * (with appropriate driver notification, if already present) 		 */		index = 0;		do {			new_func = shpchp_slot_find(ctrl->slot_bus, func->device, index++);			if (new_func && !new_func->pci_dev) {				dbg("%s:call pci_hp_configure_dev\n", __FUNCTION__);				shpchp_configure_device(ctrl, new_func);			}		} while (new_func);		/* Wait for exclusive access to hardware */		down(&ctrl->crit_sect);		p_slot->hpc_ops->green_led_on(p_slot);		/* Wait for the command to complete */		wait_for_ctrl_irq (ctrl);		/* Done with exclusive hardware access */		up(&ctrl->crit_sect);	} else {		/* Wait for exclusive access to hardware */		down(&ctrl->crit_sect);		/* turn off slot, turn on Amber LED, turn off Green LED */		rc = p_slot->hpc_ops->slot_disable(p_slot);		if (rc) {			err("%s: Issue of Slot Disable command failed\n", __FUNCTION__);			/* Done with exclusive hardware access */			up(&ctrl->crit_sect);			return rc;		}		/* Wait for the command to complete */		wait_for_ctrl_irq (ctrl);		rc = p_slot->hpc_ops->check_cmd_status(ctrl);		if (rc) {			err("%s: Failed to disable slot, error code(%d)\n", __FUNCTION__, rc);			/* Done with exclusive hardware access */			up(&ctrl->crit_sect);			return rc;  		}		/* Done with exclusive hardware access */		up(&ctrl->crit_sect);		return(rc);	}	return 0;}/** * remove_board - Turns off slot and LED's * */static u32 remove_board(struct pci_func *func, struct controller *ctrl){	int index;	u8 skip = 0;	u8 device;	u8 hp_slot;	u32 rc;

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -