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

📄 shpchp_ctrl.c

📁 优龙2410linux2.6.8内核源代码
💻 C
📖 第 1 页 / 共 5 页
字号:
					/* Done with exclusive hardware access */					up(&ctrl->crit_sect);					return WRONG_BUS_FREQUENCY;				}				/* Done with exclusive hardware access */				up(&ctrl->crit_sect);			}		}	}	/* Wait for exclusive access to hardware */	down(&ctrl->crit_sect);	/* turn on board, blink green LED, turn off Amber LED */	rc = p_slot->hpc_ops->slot_enable(p_slot);		if (rc) {		err("%s: Issue of Slot Enable 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 enable 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);	/* Wait for ~1 second */	dbg("%s: before long_delay\n", __FUNCTION__);	wait_for_ctrl_irq (ctrl);	dbg("%s: afterlong_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;	struct resource_lists res_lists;	struct pci_func *temp_func;	struct slot *p_slot;	if (func == NULL)		return(1);	if (shpchp_unconfigure_device(func))		return(1);	device = func->device;	hp_slot = func->device - ctrl->slot_device_offset;	p_slot = shpchp_find_slot(ctrl, hp_slot + ctrl->slot_device_offset);	dbg("In %s, hp_slot = %d\n", __FUNCTION__, hp_slot);	if ((ctrl->add_support) &&		!(func->bus_head || func->mem_head || func->p_mem_head || func->io_head)) {		/* Here we check to see if we've saved any of the board's		 * resources already.  If so, we'll skip the attempt to		 * determine what's being used.		 */		index = 0;		temp_func = func;		while ((temp_func = shpchp_slot_find(temp_func->bus, temp_func->device, index++))) {			if (temp_func->bus_head || temp_func->mem_head			    || temp_func->p_mem_head || temp_func->io_head) {				skip = 1;				break;			}		}		if (!skip)			rc = shpchp_save_used_resources(ctrl, func, DISABLE_CARD);	}	/* Change status to shutdown */	if (func->is_a_board)		func->status = 0x01;	func->configured = 0;	/* 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;  	}		rc = p_slot->hpc_ops->set_attention_status(p_slot, 0);	if (rc) {		err("%s: Issue of Set Attention 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);	/* Done with exclusive hardware access */	up(&ctrl->crit_sect);	if (ctrl->add_support) {		while (func) {			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;			dbg("Returning resources to ctlr lists for (B/D/F) = (%#x/%#x/%#x)\n", func->bus, 				func->device, func->function);			shpchp_return_board_resources(func, &res_lists);			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 (is_bridge(func)) {				dbg("PCI Bridge Hot-Remove s:b:d:f(%02x:%02x:%02x:%02x)\n", ctrl->seg, func->bus, 					func->device, func->function);				bridge_slot_remove(func);			} else				dbg("PCI Function Hot-Remove s:b:d:f(%02x:%02x:%02x:%02x)\n", ctrl->seg, func->bus, 					func->device, func->function);				slot_remove(func);			func = shpchp_slot_find(ctrl->slot_bus, device, 0);		}		/* Setup slot structure with entry for empty slot */		func = shpchp_slot_create(ctrl->slot_bus);		if (func == NULL) {			return(1);		}		func->bus = ctrl->slot_bus;		func->device = device;		func->function = 0;		func->configured = 0;		func->switch_save = 0x10;		func->pwr_save = 0;		func->is_a_board = 0;	}	return 0;}static void pushbutton_helper_thread (unsigned long data){	pushbutton_pending = data;	up(&event_semaphore);}/** * shpchp_pushbutton_thread * * Scheduled procedure to handle blocking stuff for the pushbuttons * Handles all pending events and exits. * */static void shpchp_pushbutton_thread (unsigned long slot){	struct slot *p_slot = (struct slot *) slot;	u8 getstatus;		pushbutton_pending = 0;	if (!p_slot) {		dbg("%s: Error! slot NULL\n", __FUNCTION__);		return;	}	p_slot->hpc_ops->get_power_status(p_slot, &getstatus);	if (getstatus) {		p_slot->state = POWEROFF_STATE;		dbg("In power_down_board, b:d(%x:%x)\n", p_slot->bus, p_slot->device);		shpchp_disable_slot(p_slot);		p_slot->state = STATIC_STATE;	} else {		p_slot->state = POWERON_STATE;		dbg("In add_board, b:d(%x:%x)\n", p_slot->bus, p_slot->device);		if (shpchp_enable_slot(p_slot)) {			/* Wait for exclusive access to hardware */			down(&p_slot->ctrl->crit_sect);			p_slot->hpc_ops->green_led_off(p_slot);			/* Wait for the command to complete */			wait_for_ctrl_irq (p_slot->ctrl);			/* Done with exclusive hardware access */			up(&p_slot->ctrl->crit_sect);		}		p_slot->state = STATIC_STATE;	}	return;}/* this is the main worker thread */static int event_thread(void* data){	struct controller *ctrl;	lock_kernel();	daemonize("shpchpd_event");	unlock_kernel();	while (1) {		dbg("!!!!event_thread sleeping\n");		down_interruptible (&event_semaphore);		dbg("event_thread woken finished = %d\n", event_finished);		if (event_finished || signal_pending(current))			break;		/* Do stuff here */		if (pushbutton_pending)			shpchp_pushbutton_thread(pushbutton_pending);		else			for (ctrl = shpchp_ctrl_list; ctrl; ctrl=ctrl->next)				interrupt_event_handler(ctrl);	}	dbg("event_thread signals exit\n");	up(&event_exit);	return 0;}int shpchp_event_start_thread (void){	int pid;	/* initialize our semaphores */	init_MUTEX_LOCKED(&event_exit);	event_finished=0;	init_MUTEX_LOCKED(&event_semaphore);	pid = kernel_thread(event_thread, NULL, 0);	if (pid < 0) {		err ("Can't start up our event thread\n");		return -1;	}	dbg("Our event thread pid = %d\n", pid);	return 0;}void shpchp_event_stop_thread (void){	event_finished = 1;	dbg("event_thread finish command given\n");	up(&event_semaphore);	dbg("wait for event_thread to exit\n");	down(&event_exit);}static int update_slot_info (struct slot *slot){	struct hotplug_slot_info *info;	int result;	info = kmalloc(sizeof(*info), GFP_KERNEL);	if (!info)		return -ENOMEM;	slot->hpc_ops->get_power_status(slot, &(info->power_status));	slot->hpc_ops->get_attention_status(slot, &(info->attention_status));	slot->hpc_ops->get_latch_status(slot, &(info->latch_status));	slot->hpc_ops->get_adapter_status(slot, &(info->adapter_status));	result = pci_hp_change_slot_info(slot->hotplug_slot, info);	kfree (info);	return result;}static void interrupt_event_handler(struct controller *ctrl){	int loop = 0;	int change = 1;	struct pci_func *func;	u8 hp_slot;	u8 getstatus;	struct slot *p_slot;	dbg("%s:\n", __FUNCTION__);	while (change) {		change = 0;		for (loop = 0; loop < 10; loop++) {			if (ctrl->event_queue[loop].event_type != 0) {				dbg("%s:loop %x event_type %x\n", __FUNCTION__, loop, 					ctrl->event_queue[loop].event_type);				hp_slot = ctrl->event_queue[loop].hp_slot;				func = shpchp_slot_find(ctrl->slot_bus, (hp_slot + ctrl->slot_device_offset), 0);				p_slot = shpchp_find_slot(ctrl, hp_slot + ctrl->slot_device_offset);				dbg("%s: hp_slot %d, func %p, p_slot %p\n", __FUNCTION__, hp_slot, func, p_slot);				if (ctrl->event_queue[loop].event_type == INT_BUTTON_CANCEL) {					dbg("%s: button cancel\n", __FUNCTION__);					del_timer(&p_slot->task_event);					switch (p_slot->state) {					case BLINKINGOFF_STATE:						/* 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);						p_slot->hpc_ops->set_attention_status(p_slot, 0);

⌨️ 快捷键说明

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