1 /****************************************************************************** 2 * 3 * Module Name: utaddress - OpRegion address range check 4 * 5 *****************************************************************************/ 6 7 /* 8 * Copyright (C) 2000 - 2014, Intel Corp. 9 * All rights reserved. 10 * 11 * Redistribution and use in source and binary forms, with or without 12 * modification, are permitted provided that the following conditions 13 * are met: 14 * 1. Redistributions of source code must retain the above copyright 15 * notice, this list of conditions, and the following disclaimer, 16 * without modification. 17 * 2. Redistributions in binary form must reproduce at minimum a disclaimer 18 * substantially similar to the "NO WARRANTY" disclaimer below 19 * ("Disclaimer") and any redistribution must be conditioned upon 20 * including a substantially similar Disclaimer requirement for further 21 * binary redistribution. 22 * 3. Neither the names of the above-listed copyright holders nor the names 23 * of any contributors may be used to endorse or promote products derived 24 * from this software without specific prior written permission. 25 * 26 * Alternatively, this software may be distributed under the terms of the 27 * GNU General Public License ("GPL") version 2 as published by the Free 28 * Software Foundation. 29 * 30 * NO WARRANTY 31 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS 32 * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT 33 * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTIBILITY AND FITNESS FOR 34 * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT 35 * HOLDERS OR CONTRIBUTORS BE LIABLE FOR SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 36 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 37 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 38 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, 39 * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING 40 * IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE 41 * POSSIBILITY OF SUCH DAMAGES. 42 */ 43 44 #define __UTADDRESS_C__ 45 46 #include <contrib/dev/acpica/include/acpi.h> 47 #include <contrib/dev/acpica/include/accommon.h> 48 #include <contrib/dev/acpica/include/acnamesp.h> 49 50 51 #define _COMPONENT ACPI_UTILITIES 52 ACPI_MODULE_NAME ("utaddress") 53 54 55 /******************************************************************************* 56 * 57 * FUNCTION: AcpiUtAddAddressRange 58 * 59 * PARAMETERS: SpaceId - Address space ID 60 * Address - OpRegion start address 61 * Length - OpRegion length 62 * RegionNode - OpRegion namespace node 63 * 64 * RETURN: Status 65 * 66 * DESCRIPTION: Add the Operation Region address range to the global list. 67 * The only supported Space IDs are Memory and I/O. Called when 68 * the OpRegion address/length operands are fully evaluated. 69 * 70 * MUTEX: Locks the namespace 71 * 72 * NOTE: Because this interface is only called when an OpRegion argument 73 * list is evaluated, there cannot be any duplicate RegionNodes. 74 * Duplicate Address/Length values are allowed, however, so that multiple 75 * address conflicts can be detected. 76 * 77 ******************************************************************************/ 78 79 ACPI_STATUS 80 AcpiUtAddAddressRange ( 81 ACPI_ADR_SPACE_TYPE SpaceId, 82 ACPI_PHYSICAL_ADDRESS Address, 83 UINT32 Length, 84 ACPI_NAMESPACE_NODE *RegionNode) 85 { 86 ACPI_ADDRESS_RANGE *RangeInfo; 87 ACPI_STATUS Status; 88 89 90 ACPI_FUNCTION_TRACE (UtAddAddressRange); 91 92 93 if ((SpaceId != ACPI_ADR_SPACE_SYSTEM_MEMORY) && 94 (SpaceId != ACPI_ADR_SPACE_SYSTEM_IO)) 95 { 96 return_ACPI_STATUS (AE_OK); 97 } 98 99 /* Allocate/init a new info block, add it to the appropriate list */ 100 101 RangeInfo = ACPI_ALLOCATE (sizeof (ACPI_ADDRESS_RANGE)); 102 if (!RangeInfo) 103 { 104 return_ACPI_STATUS (AE_NO_MEMORY); 105 } 106 107 RangeInfo->StartAddress = Address; 108 RangeInfo->EndAddress = (Address + Length - 1); 109 RangeInfo->RegionNode = RegionNode; 110 111 Status = AcpiUtAcquireMutex (ACPI_MTX_NAMESPACE); 112 if (ACPI_FAILURE (Status)) 113 { 114 ACPI_FREE (RangeInfo); 115 return_ACPI_STATUS (Status); 116 } 117 118 RangeInfo->Next = AcpiGbl_AddressRangeList[SpaceId]; 119 AcpiGbl_AddressRangeList[SpaceId] = RangeInfo; 120 121 ACPI_DEBUG_PRINT ((ACPI_DB_NAMES, 122 "\nAdded [%4.4s] address range: 0x%p-0x%p\n", 123 AcpiUtGetNodeName (RangeInfo->RegionNode), 124 ACPI_CAST_PTR (void, Address), 125 ACPI_CAST_PTR (void, RangeInfo->EndAddress))); 126 127 (void) AcpiUtReleaseMutex (ACPI_MTX_NAMESPACE); 128 return_ACPI_STATUS (AE_OK); 129 } 130 131 132 /******************************************************************************* 133 * 134 * FUNCTION: AcpiUtRemoveAddressRange 135 * 136 * PARAMETERS: SpaceId - Address space ID 137 * RegionNode - OpRegion namespace node 138 * 139 * RETURN: None 140 * 141 * DESCRIPTION: Remove the Operation Region from the global list. The only 142 * supported Space IDs are Memory and I/O. Called when an 143 * OpRegion is deleted. 144 * 145 * MUTEX: Assumes the namespace is locked 146 * 147 ******************************************************************************/ 148 149 void 150 AcpiUtRemoveAddressRange ( 151 ACPI_ADR_SPACE_TYPE SpaceId, 152 ACPI_NAMESPACE_NODE *RegionNode) 153 { 154 ACPI_ADDRESS_RANGE *RangeInfo; 155 ACPI_ADDRESS_RANGE *Prev; 156 157 158 ACPI_FUNCTION_TRACE (UtRemoveAddressRange); 159 160 161 if ((SpaceId != ACPI_ADR_SPACE_SYSTEM_MEMORY) && 162 (SpaceId != ACPI_ADR_SPACE_SYSTEM_IO)) 163 { 164 return_VOID; 165 } 166 167 /* Get the appropriate list head and check the list */ 168 169 RangeInfo = Prev = AcpiGbl_AddressRangeList[SpaceId]; 170 while (RangeInfo) 171 { 172 if (RangeInfo->RegionNode == RegionNode) 173 { 174 if (RangeInfo == Prev) /* Found at list head */ 175 { 176 AcpiGbl_AddressRangeList[SpaceId] = RangeInfo->Next; 177 } 178 else 179 { 180 Prev->Next = RangeInfo->Next; 181 } 182 183 ACPI_DEBUG_PRINT ((ACPI_DB_NAMES, 184 "\nRemoved [%4.4s] address range: 0x%p-0x%p\n", 185 AcpiUtGetNodeName (RangeInfo->RegionNode), 186 ACPI_CAST_PTR (void, RangeInfo->StartAddress), 187 ACPI_CAST_PTR (void, RangeInfo->EndAddress))); 188 189 ACPI_FREE (RangeInfo); 190 return_VOID; 191 } 192 193 Prev = RangeInfo; 194 RangeInfo = RangeInfo->Next; 195 } 196 197 return_VOID; 198 } 199 200 201 /******************************************************************************* 202 * 203 * FUNCTION: AcpiUtCheckAddressRange 204 * 205 * PARAMETERS: SpaceId - Address space ID 206 * Address - Start address 207 * Length - Length of address range 208 * Warn - TRUE if warning on overlap desired 209 * 210 * RETURN: Count of the number of conflicts detected. Zero is always 211 * returned for Space IDs other than Memory or I/O. 212 * 213 * DESCRIPTION: Check if the input address range overlaps any of the 214 * ASL operation region address ranges. The only supported 215 * Space IDs are Memory and I/O. 216 * 217 * MUTEX: Assumes the namespace is locked. 218 * 219 ******************************************************************************/ 220 221 UINT32 222 AcpiUtCheckAddressRange ( 223 ACPI_ADR_SPACE_TYPE SpaceId, 224 ACPI_PHYSICAL_ADDRESS Address, 225 UINT32 Length, 226 BOOLEAN Warn) 227 { 228 ACPI_ADDRESS_RANGE *RangeInfo; 229 ACPI_PHYSICAL_ADDRESS EndAddress; 230 char *Pathname; 231 UINT32 OverlapCount = 0; 232 233 234 ACPI_FUNCTION_TRACE (UtCheckAddressRange); 235 236 237 if ((SpaceId != ACPI_ADR_SPACE_SYSTEM_MEMORY) && 238 (SpaceId != ACPI_ADR_SPACE_SYSTEM_IO)) 239 { 240 return_UINT32 (0); 241 } 242 243 RangeInfo = AcpiGbl_AddressRangeList[SpaceId]; 244 EndAddress = Address + Length - 1; 245 246 /* Check entire list for all possible conflicts */ 247 248 while (RangeInfo) 249 { 250 /* 251 * Check if the requested address/length overlaps this 252 * address range. There are four cases to consider: 253 * 254 * 1) Input address/length is contained completely in the 255 * address range 256 * 2) Input address/length overlaps range at the range start 257 * 3) Input address/length overlaps range at the range end 258 * 4) Input address/length completely encompasses the range 259 */ 260 if ((Address <= RangeInfo->EndAddress) && 261 (EndAddress >= RangeInfo->StartAddress)) 262 { 263 /* Found an address range overlap */ 264 265 OverlapCount++; 266 if (Warn) /* Optional warning message */ 267 { 268 Pathname = AcpiNsGetExternalPathname (RangeInfo->RegionNode); 269 270 ACPI_WARNING ((AE_INFO, 271 "%s range 0x%p-0x%p conflicts with OpRegion 0x%p-0x%p (%s)", 272 AcpiUtGetRegionName (SpaceId), 273 ACPI_CAST_PTR (void, Address), 274 ACPI_CAST_PTR (void, EndAddress), 275 ACPI_CAST_PTR (void, RangeInfo->StartAddress), 276 ACPI_CAST_PTR (void, RangeInfo->EndAddress), 277 Pathname)); 278 ACPI_FREE (Pathname); 279 } 280 } 281 282 RangeInfo = RangeInfo->Next; 283 } 284 285 return_UINT32 (OverlapCount); 286 } 287 288 289 /******************************************************************************* 290 * 291 * FUNCTION: AcpiUtDeleteAddressLists 292 * 293 * PARAMETERS: None 294 * 295 * RETURN: None 296 * 297 * DESCRIPTION: Delete all global address range lists (called during 298 * subsystem shutdown). 299 * 300 ******************************************************************************/ 301 302 void 303 AcpiUtDeleteAddressLists ( 304 void) 305 { 306 ACPI_ADDRESS_RANGE *Next; 307 ACPI_ADDRESS_RANGE *RangeInfo; 308 int i; 309 310 311 /* Delete all elements in all address range lists */ 312 313 for (i = 0; i < ACPI_ADDRESS_RANGE_MAX; i++) 314 { 315 Next = AcpiGbl_AddressRangeList[i]; 316 317 while (Next) 318 { 319 RangeInfo = Next; 320 Next = RangeInfo->Next; 321 ACPI_FREE (RangeInfo); 322 } 323 324 AcpiGbl_AddressRangeList[i] = NULL; 325 } 326 } 327