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

📄 pciehp_ctrl.c

📁 底层驱动开发
💻 C
📖 第 1 页 / 共 5 页
字号:
		stop_it = 0;		/*  The following loop skips to the next present function		 *  and creates a board structure		 */		while ((function < max_functions) && (!stop_it)) {			pci_bus_read_config_dword(pci_bus, PCI_DEVFN(func->device, function), 0x00, &ID);			if (ID == 0xFFFFFFFF) {	  /* There's nothing there. */				function++;			} else {  /* There's something there */				/* Setup slot structure. */				new_slot = pciehp_slot_create(func->bus);				if (new_slot == NULL) {					/* Out of memory */					return 1;				}				new_slot->bus = func->bus;				new_slot->device = func->device;				new_slot->function = function;				new_slot->is_a_board = 1;				new_slot->status = 0;				stop_it++;			}		}	} while (function < max_functions);	dbg("returning from %s\n", __FUNCTION__);	return 0;}/* * Configuration logic that involves the hotplug data structures and  * their bookkeeping *//** * configure_bridge: fill bridge's registers, either configure or disable it. */static intconfigure_bridge(struct pci_bus *pci_bus, unsigned int devfn,			struct pci_resource *mem_node,			struct pci_resource **hold_mem_node,			int base_addr, int limit_addr){	u16 temp_word;	u32 rc;	if (mem_node) {		memcpy(*hold_mem_node, mem_node, sizeof(struct pci_resource));		mem_node->next = NULL;		/* set Mem base and Limit registers */		RES_CHECK(mem_node->base, 16);		temp_word = (u16)(mem_node->base >> 16);		rc = pci_bus_write_config_word(pci_bus, devfn, base_addr, temp_word);		RES_CHECK(mem_node->base + mem_node->length - 1, 16);		temp_word = (u16)((mem_node->base + mem_node->length - 1) >> 16);		rc = pci_bus_write_config_word(pci_bus, devfn, limit_addr, temp_word);	} else {		temp_word = 0xFFFF;		rc = pci_bus_write_config_word(pci_bus, devfn, base_addr, temp_word);		temp_word = 0x0000;		rc = pci_bus_write_config_word(pci_bus, devfn, limit_addr, temp_word);		kfree(*hold_mem_node);		*hold_mem_node = NULL;	}	return rc;}static intconfigure_new_bridge(struct controller *ctrl, struct pci_func *func,		u8 behind_bridge, struct resource_lists *resources,		struct pci_bus *pci_bus){	int cloop;	u8 temp_byte;	u8 device;	u16 temp_word;	u32 rc;	u32 ID;	unsigned int devfn;	struct pci_resource *mem_node;	struct pci_resource *p_mem_node;	struct pci_resource *io_node;	struct pci_resource *bus_node;	struct pci_resource *hold_mem_node;	struct pci_resource *hold_p_mem_node;	struct pci_resource *hold_IO_node;	struct pci_resource *hold_bus_node;	struct irq_mapping irqs;	struct pci_func *new_slot;	struct resource_lists temp_resources;	devfn = PCI_DEVFN(func->device, func->function);	/* set Primary bus */	dbg("set Primary bus = 0x%x\n", func->bus);	rc = pci_bus_write_config_byte(pci_bus, devfn, PCI_PRIMARY_BUS, func->bus);	if (rc)		return rc;	/* find range of busses to use */	bus_node = get_max_resource(&resources->bus_head, 1L);	/* If we don't have any busses to allocate, we can't continue */	if (!bus_node) {		err("Got NO bus resource to use\n");		return -ENOMEM;	}	dbg("Got ranges of buses to use: base:len=0x%x:%x\n", bus_node->base, bus_node->length);	/* set Secondary bus */	temp_byte = (u8)bus_node->base;	dbg("set Secondary bus = 0x%x\n", temp_byte);	rc = pci_bus_write_config_byte(pci_bus, devfn, PCI_SECONDARY_BUS, temp_byte);	if (rc)		return rc;	/* set subordinate bus */	temp_byte = (u8)(bus_node->base + bus_node->length - 1);	dbg("set subordinate bus = 0x%x\n", temp_byte);	rc = pci_bus_write_config_byte (pci_bus, devfn, PCI_SUBORDINATE_BUS, temp_byte);	if (rc)		return rc;	/* Set HP parameters (Cache Line Size, Latency Timer) */	rc = pciehprm_set_hpp(ctrl, func, PCI_HEADER_TYPE_BRIDGE);	if (rc)		return rc;	/* Setup the IO, memory, and prefetchable windows */	io_node = get_max_resource(&(resources->io_head), 0x1000L);	if (io_node) {		dbg("io_node(base, len, next) (%x, %x, %p)\n", io_node->base,				io_node->length, io_node->next);	}	mem_node = get_max_resource(&(resources->mem_head), 0x100000L);	if (mem_node) {		dbg("mem_node(base, len, next) (%x, %x, %p)\n", mem_node->base,				mem_node->length, mem_node->next);	}	if (resources->p_mem_head)		p_mem_node = get_max_resource(&(resources->p_mem_head), 0x100000L);	else {		/*		 * In some platform implementation, MEM and PMEM are not		 *  distinguished, and hence ACPI _CRS has only MEM entries		 *  for both MEM and PMEM.		 */		dbg("using MEM for PMEM\n");		p_mem_node = get_max_resource(&(resources->mem_head), 0x100000L);	}	if (p_mem_node) {		dbg("p_mem_node(base, len, next) (%x, %x, %p)\n", p_mem_node->base,				p_mem_node->length, p_mem_node->next);	}	/* set up the IRQ info */	if (!resources->irqs) {		irqs.barber_pole = 0;		irqs.interrupt[0] = 0;		irqs.interrupt[1] = 0;		irqs.interrupt[2] = 0;		irqs.interrupt[3] = 0;		irqs.valid_INT = 0;	} else {		irqs.barber_pole = resources->irqs->barber_pole;		irqs.interrupt[0] = resources->irqs->interrupt[0];		irqs.interrupt[1] = resources->irqs->interrupt[1];		irqs.interrupt[2] = resources->irqs->interrupt[2];		irqs.interrupt[3] = resources->irqs->interrupt[3];		irqs.valid_INT = resources->irqs->valid_INT;	}	/* set up resource lists that are now aligned on top and bottom	 * for anything behind the bridge.	 */	temp_resources.bus_head = bus_node;	temp_resources.io_head = io_node;	temp_resources.mem_head = mem_node;	temp_resources.p_mem_head = p_mem_node;	temp_resources.irqs = &irqs;	/* Make copies of the nodes we are going to pass down so that	 * if there is a problem,we can just use these to free resources	 */	hold_bus_node = kmalloc(sizeof(struct pci_resource), GFP_KERNEL);	hold_IO_node = kmalloc(sizeof(struct pci_resource), GFP_KERNEL);	hold_mem_node = kmalloc(sizeof(struct pci_resource), GFP_KERNEL);	hold_p_mem_node = kmalloc(sizeof(struct pci_resource), GFP_KERNEL);	if (!hold_bus_node || !hold_IO_node || !hold_mem_node || !hold_p_mem_node) {		kfree(hold_bus_node);		kfree(hold_IO_node);		kfree(hold_mem_node);		kfree(hold_p_mem_node);		return 1;	}	memcpy(hold_bus_node, bus_node, sizeof(struct pci_resource));	bus_node->base += 1;	bus_node->length -= 1;	bus_node->next = NULL;	/* If we have IO resources copy them and fill in the bridge's	 * IO range registers	 */	if (io_node) {		memcpy(hold_IO_node, io_node, sizeof(struct pci_resource));		io_node->next = NULL;		/* set IO base and Limit registers */		RES_CHECK(io_node->base, 8);		temp_byte = (u8)(io_node->base >> 8);		rc = pci_bus_write_config_byte (pci_bus, devfn, PCI_IO_BASE, temp_byte);		RES_CHECK(io_node->base + io_node->length - 1, 8);		temp_byte = (u8)((io_node->base + io_node->length - 1) >> 8);		rc = pci_bus_write_config_byte (pci_bus, devfn, PCI_IO_LIMIT, temp_byte);	} else {		kfree(hold_IO_node);		hold_IO_node = NULL;	}	/* If we have memory resources copy them and fill in the bridge's	 * memory range registers.  Otherwise, fill in the range	 * registers with values that disable them.	 */	rc = configure_bridge(pci_bus, devfn, mem_node, &hold_mem_node,				PCI_MEMORY_BASE, PCI_MEMORY_LIMIT);	/* If we have prefetchable memory resources copy them and 	 * fill in the bridge's memory range registers.  Otherwise,	 * fill in the range registers with values that disable them.	 */	rc = configure_bridge(pci_bus, devfn, p_mem_node, &hold_p_mem_node,				PCI_PREF_MEMORY_BASE, PCI_PREF_MEMORY_LIMIT);	/* Adjust this to compensate for extra adjustment in first loop */	irqs.barber_pole--;	rc = 0;	/* Here we actually find the devices and configure them */	for (device = 0; (device <= 0x1F) && !rc; device++) {		irqs.barber_pole = (irqs.barber_pole + 1) & 0x03;		ID = 0xFFFFFFFF;		pci_bus->number = hold_bus_node->base;		pci_bus_read_config_dword (pci_bus, PCI_DEVFN(device, 0), PCI_VENDOR_ID, &ID);		pci_bus->number = func->bus;		if (ID != 0xFFFFFFFF) {	  /*  device Present */			/* Setup slot structure. */			new_slot = pciehp_slot_create(hold_bus_node->base);			if (new_slot == NULL) {				/* Out of memory */				rc = -ENOMEM;				continue;			}			new_slot->bus = hold_bus_node->base;			new_slot->device = device;			new_slot->function = 0;			new_slot->is_a_board = 1;			new_slot->status = 0;			rc = configure_new_device(ctrl, new_slot, 1,					&temp_resources, func->bus,					func->device);			dbg("configure_new_device rc=0x%x\n",rc);		}	/* End of IF (device in slot?) */	}		/* End of FOR loop */	if (rc) {		pciehp_destroy_resource_list(&temp_resources);		return_resource(&(resources->bus_head), hold_bus_node);		return_resource(&(resources->io_head), hold_IO_node);		return_resource(&(resources->mem_head), hold_mem_node);		return_resource(&(resources->p_mem_head), hold_p_mem_node);		return(rc);	}	/* save the interrupt routing information */	if (resources->irqs) {		resources->irqs->interrupt[0] = irqs.interrupt[0];		resources->irqs->interrupt[1] = irqs.interrupt[1];		resources->irqs->interrupt[2] = irqs.interrupt[2];		resources->irqs->interrupt[3] = irqs.interrupt[3];		resources->irqs->valid_INT = irqs.valid_INT;	} else if (!behind_bridge) {		/* We need to hook up the interrupts here */		for (cloop = 0; cloop < 4; cloop++) {			if (irqs.valid_INT & (0x01 << cloop)) {				rc = pciehp_set_irq(func->bus, func->device,							0x0A + cloop, irqs.interrupt[cloop]);				if (rc) {					pciehp_destroy_resource_list (&temp_resources);					return_resource(&(resources->bus_head), hold_bus_node);					return_resource(&(resources->io_head), hold_IO_node);					return_resource(&(resources->mem_head), hold_mem_node);					return_resource(&(resources->p_mem_head), hold_p_mem_node);					return rc;				}			}		}	/* end of for loop */	}	/* Return unused bus resources	 * First use the temporary node to store information for the board	 */	if (hold_bus_node && bus_node && temp_resources.bus_head) {		hold_bus_node->length = bus_node->base - hold_bus_node->base;		hold_bus_node->next = func->bus_head;		func->bus_head = hold_bus_node;		temp_byte = (u8)(temp_resources.bus_head->base - 1);		/* set subordinate bus */		dbg("re-set subordinate bus = 0x%x\n", temp_byte);		rc = pci_bus_write_config_byte (pci_bus, devfn, PCI_SUBORDINATE_BUS, temp_byte);		if (temp_resources.bus_head->length == 0) {			kfree(temp_resources.bus_head);			temp_resources.bus_head = NULL;		} else {			dbg("return bus res of b:d(0x%x:%x) base:len(0x%x:%x)\n",				func->bus, func->device, temp_resources.bus_head->base, temp_resources.bus_head->length);			return_resource(&(resources->bus_head), temp_resources.bus_head);		}	}	/* If we have IO space available and there is some left,	 * return the unused portion	 */	if (hold_IO_node && temp_resources.io_head) {		io_node = do_pre_bridge_resource_split(&(temp_resources.io_head),							&hold_IO_node, 0x1000);		/* Check if we were able to split something off */		if (io_node) {			hold_IO_node->base = io_node->base + io_node->length;			RES_CHECK(hold_IO_node->base, 8);			temp_byte = (u8)((hold_IO_node->base) >> 8);			rc = pci_bus_write_config_byte (pci_bus, devfn, PCI_IO_BASE, temp_byte);			return_resource(&(resources->io_head), io_node);		}		io_node = do_bridge_resource_split(&(temp_resources.io_head), 0x1000);		/*  Check if we were able to split something off */		if (io_node) {			/* First use the temporary node to store information for the board */			hold_IO_node->length = io_node->base - hold_IO_node->base;			/* If we used any, add it to the board's list */			if (hold_IO_node->length) {				hold_IO_node->next = func->io_head;				func->io_head = hold_IO_node;				RES_CHECK(io_node->base - 1, 8);				temp_byte = (u8)((io_node->base - 1) >> 8);				rc = pci_bus_write_config_byte (pci_bus, devfn, PCI_IO_LIMIT, temp_byte);				return_resource(&(resources->io_head), io_node);			} else {				/* it doesn't need any IO */				temp_byte = 0x00;				rc = pci_bus_write_config_byte(pci_bus, devfn, PCI_IO_LIMIT, temp_byte);				return_resource(&(resources->io_head), io_node);				kfree(hold_IO_node);			}		} else {			/* it used most of the range */			hold_IO_node->next = func->io_head;			func->io_head = hold_IO_node;		}	} else if (hold_IO_node) {		/* it used the whole range */		hold_IO_node->next = func->io_head;		func->io_head = hold_IO_node;	}	/* If we have memory space available and there is some left,	 * return the unused portion	 */	if (hold_mem_node && temp_resources.mem_head) {		mem_node = do_pre_bridge_resource_split(&(temp_resources.mem_head), &hold_mem_node, 0x100000L);		/* Check if we were able to split something off */		if (mem_node) {			hold_mem_node->base = mem_node->base + mem_node->length;			RES_CHECK(hold_mem_node->base, 16);			temp_word = (u16)((hold_mem_node->base) >> 16);			rc = pci_bus_write_config_word (pci_bus, devfn, PCI_MEMORY_BASE, temp_word);			return_resource(&(resources->mem_head), mem_node);		}		mem_node = do_bridge_resource_split(&(temp_resources.mem_head), 0x100000L);		/* Check if we were able to split something off */		if (mem_node) {			/* First use the temporary node to store information for the board */			hold_mem_node->length = mem_node->base - hold_mem_node->base;			if (hold_mem_node->length) {				hold_mem_node->next = func->mem_head;				func->mem_head = hold_mem_node;				/* configure end address */				RES_CHECK(mem_node->base - 1, 16);				temp_word = (u16)((mem_node->base - 1) >> 16);				rc = pci_bus_write_config_word (pci_bus, devfn, PCI_MEMORY_LIMIT, temp_word);				/* Return unused resources to the pool */				return_resource(&(resources->mem_head), mem_node);			} else {				/* it doesn't need any Mem */				temp_word = 0x0000;				rc = pci_bus_write_config_word (pci_bus, devfn, PCI_MEMORY_LIMIT, temp_word);				return_resource(&(resources->mem_head), mem_node);				kfree(hold_mem_node);			}		} else {			/* it used most of the range */			hold_mem_node->next = func->mem_head;			func->mem_head = hold_mem_node;		}	} else if (hold_mem_node) {		/* it used the whole range */		hold_mem_node->next = func->mem_head;		func->mem_head = hold_mem_node;	}	/* If we have prefetchable memory space available and there is some 	 * left at the end, return the unused portion	 */	if (hold_p_mem_node && temp_resources.p_mem_head) {		p_mem_node = do_pre_bridge_resource_split(&(temp_resources.p_mem_head),								&hold_p_mem_node, 0x100000L);		/* Check if we were able to split something off */		if (p_mem_node) {			hold_p_mem_node->base = p_mem_node->base + p_mem_node->length;			RES_CHECK(hold_p_mem_node->base, 16);			temp_word = (u16)((hold_p_mem_node->base) >> 16);			rc = pci_bus_write_config_word (pci_bus, devfn, PCI_PREF_MEMORY_BASE, temp_word);			return_resource(&(resources->p_mem_head), p_mem_node);		}		p_mem_node = do_bridge_resource_split(&(temp_resources.p_mem_head), 0x100000L);		/* Check if we were able to split something off */		if (p_mem_node) {			/* First use the temporary node to store information for the board */			hold_p_mem_node->length = p_mem_node->base - hold_p_mem_node->base;			/* If we used any, add it to the board's list */			if (hold_p_mem_node->length) {				hold_p_mem_node->next = func->p_mem_head;				func->p_mem_head = hold_p_mem_node;				RES_CHECK(p_mem_node->base - 1, 16);				temp_word = (u16)((p_mem_node->base - 1) >> 16);				rc = pci_bus_write_config_word (pci_bus, devfn, PCI_PREF_MEMORY_LIMIT, temp_word);				return_resource(&(resources->p_mem_head), p_mem_no

⌨️ 快捷键说明

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