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

📄 shpchp_ctrl.c

📁 linux-2.6.15.6
💻 C
📖 第 1 页 / 共 2 页
字号:
	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 */	wait_for_ctrl_irq (ctrl);	dbg("%s: slot status = %x\n", __FUNCTION__, p_slot->status);	/* Check for a power fault */	if (p_slot->status == 0xFF) {		/* power fault occurred, but it was benign */		dbg("%s: power fault\n", __FUNCTION__);		rc = POWER_FAILURE;		p_slot->status = 0;		goto err_exit;	}	if (shpchp_configure_device(p_slot)) {		err("Cannot add device at 0x%x:0x%x\n", p_slot->bus,				p_slot->device);		goto err_exit;	}	p_slot->status = 0;	p_slot->is_a_board = 0x01;	p_slot->pwr_save = 1;	/* 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);	return 0;err_exit:	/* 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);}/** * remove_board - Turns off slot and LED's * */static int remove_board(struct slot *p_slot){	struct controller *ctrl = p_slot->ctrl;	u8 hp_slot;	int rc;	if (shpchp_unconfigure_device(p_slot))		return(1);	hp_slot = p_slot->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);	/* Change status to shutdown */	if (p_slot->is_a_board)		p_slot->status = 0x01;	/* 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);	p_slot->pwr_save = 0;	p_slot->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;		shpchp_disable_slot(p_slot);		p_slot->state = STATIC_STATE;	} else {		p_slot->state = POWERON_STATE;		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;	}	return 0;}void shpchp_event_stop_thread (void){	event_finished = 1;	up(&event_semaphore);	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;	u8 hp_slot;	u8 getstatus;	struct slot *p_slot;	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;				p_slot = shpchp_find_slot(ctrl, hp_slot + ctrl->slot_device_offset);				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);						/* Wait for the command to complete */						wait_for_ctrl_irq (ctrl);						/* Done with exclusive hardware access */						up(&ctrl->crit_sect);						break;					case BLINKINGON_STATE:						/* Wait for exclusive access to hardware */						down(&ctrl->crit_sect);						p_slot->hpc_ops->green_led_off(p_slot);						/* Wait for the command to complete */						wait_for_ctrl_irq (ctrl);						p_slot->hpc_ops->set_attention_status(p_slot, 0);						/* Wait for the command to complete */						wait_for_ctrl_irq (ctrl);						/* Done with exclusive hardware access */						up(&ctrl->crit_sect);						break;					default:						warn("Not a valid state\n");						return;					}					info(msg_button_cancel, p_slot->number);					p_slot->state = STATIC_STATE;				} else if (ctrl->event_queue[loop].event_type == INT_BUTTON_PRESS) {					/* Button Pressed (No action on 1st press...) */					dbg("%s: Button pressed\n", __FUNCTION__);					p_slot->hpc_ops->get_power_status(p_slot, &getstatus);					if (getstatus) {						/* slot is on */						dbg("%s: slot is on\n", __FUNCTION__);						p_slot->state = BLINKINGOFF_STATE;						info(msg_button_off, p_slot->number);					} else {						/* slot is off */						dbg("%s: slot is off\n", __FUNCTION__);						p_slot->state = BLINKINGON_STATE;						info(msg_button_on, p_slot->number);					}					/* Wait for exclusive access to hardware */					down(&ctrl->crit_sect);					/* blink green LED and turn off amber */					p_slot->hpc_ops->green_led_blink(p_slot);					/* Wait for the command to complete */					wait_for_ctrl_irq (ctrl);										p_slot->hpc_ops->set_attention_status(p_slot, 0);					/* Wait for the command to complete */					wait_for_ctrl_irq (ctrl);					/* Done with exclusive hardware access */					up(&ctrl->crit_sect);					init_timer(&p_slot->task_event);					p_slot->task_event.expires = jiffies + 5 * HZ;   /* 5 second delay */					p_slot->task_event.function = (void (*)(unsigned long)) pushbutton_helper_thread;					p_slot->task_event.data = (unsigned long) p_slot;					dbg("%s: add_timer p_slot = %p\n", __FUNCTION__,(void *) p_slot);					add_timer(&p_slot->task_event);				} else if (ctrl->event_queue[loop].event_type == INT_POWER_FAULT) {					/***********POWER FAULT********************/					dbg("%s: power fault\n", __FUNCTION__);					/* Wait for exclusive access to hardware */					down(&ctrl->crit_sect);					p_slot->hpc_ops->set_attention_status(p_slot, 1);					/* Wait for the command to complete */					wait_for_ctrl_irq (ctrl);										p_slot->hpc_ops->green_led_off(p_slot);					/* Wait for the command to complete */					wait_for_ctrl_irq (ctrl);					/* Done with exclusive hardware access */					up(&ctrl->crit_sect);				} else {					/* refresh notification */					if (p_slot)						update_slot_info(p_slot);				}				ctrl->event_queue[loop].event_type = 0;				change = 1;			}		}		/* End of FOR loop */	}	return;}int shpchp_enable_slot (struct slot *p_slot){	u8 getstatus = 0;	int rc;	/* Check to see if (latch closed, card present, power off) */	down(&p_slot->ctrl->crit_sect);	rc = p_slot->hpc_ops->get_adapter_status(p_slot, &getstatus);	if (rc || !getstatus) {		info("%s: no adapter on slot(%x)\n", __FUNCTION__, p_slot->number);		up(&p_slot->ctrl->crit_sect);		return -ENODEV;	}	rc = p_slot->hpc_ops->get_latch_status(p_slot, &getstatus);	if (rc || getstatus) {		info("%s: latch open on slot(%x)\n", __FUNCTION__, p_slot->number);		up(&p_slot->ctrl->crit_sect);		return -ENODEV;	}	rc = p_slot->hpc_ops->get_power_status(p_slot, &getstatus);	if (rc || getstatus) {		info("%s: already enabled on slot(%x)\n", __FUNCTION__, p_slot->number);		up(&p_slot->ctrl->crit_sect);		return -ENODEV;	}	up(&p_slot->ctrl->crit_sect);	p_slot->is_a_board = 1;	/* We have to save the presence info for these slots */	p_slot->hpc_ops->get_adapter_status(p_slot, &(p_slot->presence_save));	p_slot->hpc_ops->get_power_status(p_slot, &(p_slot->pwr_save));	dbg("%s: p_slot->pwr_save %x\n", __FUNCTION__, p_slot->pwr_save);	p_slot->hpc_ops->get_latch_status(p_slot, &getstatus);	rc = board_added(p_slot);	if (rc) {		p_slot->hpc_ops->get_adapter_status(p_slot,				&(p_slot->presence_save));		p_slot->hpc_ops->get_latch_status(p_slot, &getstatus);	}	update_slot_info(p_slot);	return rc;}int shpchp_disable_slot (struct slot *p_slot){	u8 getstatus = 0;	int ret = 0;	if (!p_slot->ctrl)		return -ENODEV;	/* Check to see if (latch closed, card present, power on) */	down(&p_slot->ctrl->crit_sect);	ret = p_slot->hpc_ops->get_adapter_status(p_slot, &getstatus);	if (ret || !getstatus) {		info("%s: no adapter on slot(%x)\n", __FUNCTION__, p_slot->number);		up(&p_slot->ctrl->crit_sect);		return -ENODEV;	}	ret = p_slot->hpc_ops->get_latch_status(p_slot, &getstatus);	if (ret || getstatus) {		info("%s: latch open on slot(%x)\n", __FUNCTION__, p_slot->number);		up(&p_slot->ctrl->crit_sect);		return -ENODEV;	}	ret = p_slot->hpc_ops->get_power_status(p_slot, &getstatus);	if (ret || !getstatus) {		info("%s: already disabled slot(%x)\n", __FUNCTION__, p_slot->number);		up(&p_slot->ctrl->crit_sect);		return -ENODEV;	}	up(&p_slot->ctrl->crit_sect);	ret = remove_board(p_slot);	update_slot_info(p_slot);	return ret;}

⌨️ 快捷键说明

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