]> git.kernelconcepts.de Git - karo-tx-linux.git/commitdiff
Merge branch 'acpi-lpss'
authorRafael J. Wysocki <rafael.j.wysocki@intel.com>
Mon, 11 Feb 2013 12:21:27 +0000 (13:21 +0100)
committerRafael J. Wysocki <rafael.j.wysocki@intel.com>
Mon, 11 Feb 2013 12:21:27 +0000 (13:21 +0100)
* acpi-lpss:
  ACPI / platform: create LPSS clocks if Lynxpoint devices are found during scan
  clk: x86: add support for Lynxpoint LPSS clocks
  x86: add support for Intel Low Power Subsystem
  ACPI / platform: fix comment about the platform device name
  ACPI: add support for CSRT table

1  2 
arch/x86/Kconfig
drivers/acpi/internal.h
drivers/acpi/scan.c

diff --combined arch/x86/Kconfig
index 225543bf45a5ca551f9609b18cf8711f729f71e7,c8c9b1403f17de5b8b5f7c7b41e647528b2b4976..4f7c2da2f9f851a18c29249a2a5e0b16e0bf251a
@@@ -454,6 -454,16 +454,16 @@@ config X86_MDFL
  
  endif
  
+ config X86_INTEL_LPSS
+       bool "Intel Low Power Subsystem Support"
+       depends on ACPI
+       select COMMON_CLK
+       ---help---
+         Select to build support for Intel Low Power Subsystem such as
+         found on Intel Lynxpoint PCH. Selecting this option enables
+         things like clock tree (common clock framework) which are needed
+         by the LPSS peripheral drivers.
  config X86_RDC321X
        bool "RDC R-321x SoC"
        depends on X86_32
@@@ -2138,7 -2148,6 +2148,7 @@@ config OLPC_XO1_RT
  config OLPC_XO1_SCI
        bool "OLPC XO-1 SCI extras"
        depends on OLPC && OLPC_XO1_PM
 +      depends on INPUT=y
        select POWER_SUPPLY
        select GPIO_CS5535
        select MFD_CORE
diff --combined drivers/acpi/internal.h
index c5a61cd6c1a5200c5297f4f3260c23ee185140e7,4b68373473de34c9f6d33c19039a2ff9bb5f33da..da233477d26022157dd25a1ed1f28e99a261ea10
@@@ -26,6 -26,7 +26,7 @@@
  int init_acpi_device_notify(void);
  int acpi_scan_init(void);
  int acpi_sysfs_init(void);
+ void acpi_csrt_init(void);
  
  #ifdef CONFIG_DEBUG_FS
  extern struct dentry *acpi_debugfs_dir;
@@@ -34,34 -35,16 +35,34 @@@ int acpi_debugfs_init(void)
  static inline void acpi_debugfs_init(void) { return; }
  #endif
  
 +/* --------------------------------------------------------------------------
 +                     Device Node Initialization / Removal
 +   -------------------------------------------------------------------------- */
 +#define ACPI_STA_DEFAULT (ACPI_STA_DEVICE_PRESENT | ACPI_STA_DEVICE_ENABLED | \
 +                        ACPI_STA_DEVICE_UI | ACPI_STA_DEVICE_FUNCTIONING)
 +
 +int acpi_device_add(struct acpi_device *device,
 +                  void (*release)(struct device *));
 +void acpi_init_device_object(struct acpi_device *device, acpi_handle handle,
 +                           int type, unsigned long long sta);
 +void acpi_device_add_finalize(struct acpi_device *device);
 +void acpi_free_ids(struct acpi_device *device);
 +
  /* --------------------------------------------------------------------------
                                    Power Resource
     -------------------------------------------------------------------------- */
  int acpi_power_init(void);
 +void acpi_power_resources_list_free(struct list_head *list);
 +int acpi_extract_power_resources(union acpi_object *package, unsigned int start,
 +                               struct list_head *list);
 +int acpi_add_power_resource(acpi_handle handle);
 +void acpi_power_add_remove_device(struct acpi_device *adev, bool add);
 +int acpi_power_min_system_level(struct list_head *list);
  int acpi_device_sleep_wake(struct acpi_device *dev,
                             int enable, int sleep_state, int dev_state);
  int acpi_power_get_inferred_state(struct acpi_device *device, int *state);
  int acpi_power_on_resources(struct acpi_device *device, int state);
  int acpi_power_transition(struct acpi_device *device, int state);
 -int acpi_bus_init_power(struct acpi_device *device);
  
  int acpi_wakeup_device_init(void);
  void acpi_early_processor_set_pdc(void);
@@@ -117,6 -100,10 +118,10 @@@ static inline void suspend_nvs_restore(
    -------------------------------------------------------------------------- */
  struct platform_device;
  
- struct platform_device *acpi_create_platform_device(struct acpi_device *adev);
+ /* Flags for acpi_create_platform_device */
+ #define ACPI_PLATFORM_CLK     BIT(0)
+ struct platform_device *acpi_create_platform_device(struct acpi_device *adev,
+                                                   unsigned long flags);
  
  #endif /* _ACPI_INTERNAL_H_ */
diff --combined drivers/acpi/scan.c
index cbf6e7729c3972813b1815fa2716b87bf86510b5,fe171aa7a98c3df20b68a638fd96daf177df1847..c4358716aadcca510a58ccfb5b0d554d159b917c
@@@ -38,21 -38,20 +38,21 @@@ static const struct acpi_device_id acpi
        { "PNP0D40" },
  
        /* Haswell LPSS devices */
-       { "INT33C0", 0 },
-       { "INT33C1", 0 },
-       { "INT33C2", 0 },
-       { "INT33C3", 0 },
-       { "INT33C4", 0 },
-       { "INT33C5", 0 },
-       { "INT33C6", 0 },
-       { "INT33C7", 0 },
+       { "INT33C0", ACPI_PLATFORM_CLK },
+       { "INT33C1", ACPI_PLATFORM_CLK },
+       { "INT33C2", ACPI_PLATFORM_CLK },
+       { "INT33C3", ACPI_PLATFORM_CLK },
+       { "INT33C4", ACPI_PLATFORM_CLK },
+       { "INT33C5", ACPI_PLATFORM_CLK },
+       { "INT33C6", ACPI_PLATFORM_CLK },
+       { "INT33C7", ACPI_PLATFORM_CLK },
  
        { }
  };
  
  static LIST_HEAD(acpi_device_list);
  static LIST_HEAD(acpi_bus_id_list);
 +static DEFINE_MUTEX(acpi_scan_lock);
  DEFINE_MUTEX(acpi_device_lock);
  LIST_HEAD(acpi_wakeup_device_list);
  
@@@ -128,8 -127,13 +128,8 @@@ void acpi_bus_hot_remove_device(void *c
        ACPI_DEBUG_PRINT((ACPI_DB_INFO,
                "Hot-removing device %s...\n", dev_name(&device->dev)));
  
 -      if (acpi_bus_trim(device)) {
 -              printk(KERN_ERR PREFIX
 -                              "Removing device failed\n");
 -              goto err_out;
 -      }
 -
 -      /* device has been freed */
 +      acpi_bus_trim(device);
 +      /* Device node has been released. */
        device = NULL;
  
        /* power off device */
@@@ -174,32 -178,6 +174,32 @@@ err_out
  }
  EXPORT_SYMBOL(acpi_bus_hot_remove_device);
  
 +static ssize_t real_power_state_show(struct device *dev,
 +                                   struct device_attribute *attr, char *buf)
 +{
 +      struct acpi_device *adev = to_acpi_device(dev);
 +      int state;
 +      int ret;
 +
 +      ret = acpi_device_get_power(adev, &state);
 +      if (ret)
 +              return ret;
 +
 +      return sprintf(buf, "%s\n", acpi_power_state_string(state));
 +}
 +
 +static DEVICE_ATTR(real_power_state, 0444, real_power_state_show, NULL);
 +
 +static ssize_t power_state_show(struct device *dev,
 +                              struct device_attribute *attr, char *buf)
 +{
 +      struct acpi_device *adev = to_acpi_device(dev);
 +
 +      return sprintf(buf, "%s\n", acpi_power_state_string(adev->power.state));
 +}
 +
 +static DEVICE_ATTR(power_state, 0444, power_state_show, NULL);
 +
  static ssize_t
  acpi_eject_store(struct device *d, struct device_attribute *attr,
                const char *buf, size_t count)
@@@ -391,22 -369,8 +391,22 @@@ static int acpi_device_setup_files(stru
           * hot-removal function from userland.
           */
        status = acpi_get_handle(dev->handle, "_EJ0", &temp);
 -      if (ACPI_SUCCESS(status))
 +      if (ACPI_SUCCESS(status)) {
                result = device_create_file(&dev->dev, &dev_attr_eject);
 +              if (result)
 +                      return result;
 +      }
 +
 +      if (dev->flags.power_manageable) {
 +              result = device_create_file(&dev->dev, &dev_attr_power_state);
 +              if (result)
 +                      return result;
 +
 +              if (dev->power.flags.power_resources)
 +                      result = device_create_file(&dev->dev,
 +                                                  &dev_attr_real_power_state);
 +      }
 +
  end:
        return result;
  }
@@@ -416,13 -380,6 +416,13 @@@ static void acpi_device_remove_files(st
        acpi_status status;
        acpi_handle temp;
  
 +      if (dev->flags.power_manageable) {
 +              device_remove_file(&dev->dev, &dev_attr_power_state);
 +              if (dev->power.flags.power_resources)
 +                      device_remove_file(&dev->dev,
 +                                         &dev_attr_real_power_state);
 +      }
 +
        /*
         * If device has _STR, remove 'description' file
         */
@@@ -491,9 -448,9 +491,9 @@@ const struct acpi_device_id *acpi_match
                                               const struct device *dev)
  {
        struct acpi_device *adev;
 +      acpi_handle handle = ACPI_HANDLE(dev);
  
 -      if (!ids || !ACPI_HANDLE(dev)
 -          || ACPI_FAILURE(acpi_bus_get_device(ACPI_HANDLE(dev), &adev)))
 +      if (!ids || !handle || acpi_bus_get_device(handle, &adev))
                return NULL;
  
        return __acpi_match_device(adev, ids);
@@@ -507,7 -464,7 +507,7 @@@ int acpi_match_device_ids(struct acpi_d
  }
  EXPORT_SYMBOL(acpi_match_device_ids);
  
 -static void acpi_free_ids(struct acpi_device *device)
 +void acpi_free_ids(struct acpi_device *device)
  {
        struct acpi_hardware_id *id, *tmp;
  
                kfree(id->id);
                kfree(id);
        }
 +      kfree(device->pnp.unique_id);
 +}
 +
 +static void acpi_free_power_resources_lists(struct acpi_device *device)
 +{
 +      int i;
 +
 +      if (device->wakeup.flags.valid)
 +              acpi_power_resources_list_free(&device->wakeup.resources);
 +
 +      if (!device->flags.power_manageable)
 +              return;
 +
 +      for (i = ACPI_STATE_D0; i <= ACPI_STATE_D3_HOT; i++) {
 +              struct acpi_device_power_state *ps = &device->power.states[i];
 +              acpi_power_resources_list_free(&ps->resources);
 +      }
  }
  
  static void acpi_device_release(struct device *dev)
        struct acpi_device *acpi_dev = to_acpi_device(dev);
  
        acpi_free_ids(acpi_dev);
 -      kfree(acpi_dev->pnp.unique_id);
 +      acpi_free_power_resources_lists(acpi_dev);
        kfree(acpi_dev);
  }
  
@@@ -677,25 -617,12 +677,25 @@@ struct bus_type acpi_bus_type = 
        .uevent         = acpi_device_uevent,
  };
  
 -static int acpi_device_register(struct acpi_device *device)
 +int acpi_device_add(struct acpi_device *device,
 +                  void (*release)(struct device *))
  {
        int result;
        struct acpi_device_bus_id *acpi_device_bus_id, *new_bus_id;
        int found = 0;
  
 +      if (device->handle) {
 +              acpi_status status;
 +
 +              status = acpi_attach_data(device->handle, acpi_bus_data_handler,
 +                                        device);
 +              if (ACPI_FAILURE(status)) {
 +                      acpi_handle_err(device->handle,
 +                                      "Unable to attach device data\n");
 +                      return -ENODEV;
 +              }
 +      }
 +
        /*
         * Linkage
         * -------
        INIT_LIST_HEAD(&device->wakeup_list);
        INIT_LIST_HEAD(&device->physical_node_list);
        mutex_init(&device->physical_node_lock);
 +      INIT_LIST_HEAD(&device->power_dependent);
  
        new_bus_id = kzalloc(sizeof(struct acpi_device_bus_id), GFP_KERNEL);
        if (!new_bus_id) {
 -              printk(KERN_ERR PREFIX "Memory allocation error\n");
 -              return -ENOMEM;
 +              pr_err(PREFIX "Memory allocation error\n");
 +              result = -ENOMEM;
 +              goto err_detach;
        }
  
        mutex_lock(&acpi_device_lock);
        if (device->parent)
                device->dev.parent = &device->parent->dev;
        device->dev.bus = &acpi_bus_type;
 -      device->dev.release = &acpi_device_release;
 -      result = device_register(&device->dev);
 +      device->dev.release = release;
 +      result = device_add(&device->dev);
        if (result) {
                dev_err(&device->dev, "Error registering device\n");
 -              goto end;
 +              goto err;
        }
  
        result = acpi_device_setup_files(device);
  
        device->removal_type = ACPI_BUS_REMOVAL_NORMAL;
        return 0;
 -end:
 +
 + err:
        mutex_lock(&acpi_device_lock);
        if (device->parent)
                list_del(&device->node);
        list_del(&device->wakeup_list);
        mutex_unlock(&acpi_device_lock);
 +
 + err_detach:
 +      acpi_detach_data(device->handle, acpi_bus_data_handler);
        return result;
  }
  
@@@ -785,18 -706,8 +785,18 @@@ static void acpi_device_unregister(stru
  
        acpi_detach_data(device->handle, acpi_bus_data_handler);
  
 +      acpi_power_add_remove_device(device, false);
        acpi_device_remove_files(device);
 -      device_unregister(&device->dev);
 +      if (device->remove)
 +              device->remove(device);
 +
 +      device_del(&device->dev);
 +      /*
 +       * Drop the reference counts of all power resources the device depends
 +       * on and turn off the ones that have no more references.
 +       */
 +      acpi_power_transition(device, ACPI_STATE_D3_COLD);
 +      put_device(&device->dev);
  }
  
  /* --------------------------------------------------------------------------
@@@ -939,43 -850,52 +939,43 @@@ void acpi_bus_data_handler(acpi_handle 
        return;
  }
  
 -static int acpi_bus_get_perf_flags(struct acpi_device *device)
 -{
 -      device->performance.state = ACPI_STATE_UNKNOWN;
 -      return 0;
 -}
 -
 -static acpi_status
 -acpi_bus_extract_wakeup_device_power_package(acpi_handle handle,
 -                                           struct acpi_device_wakeup *wakeup)
 +static int acpi_bus_extract_wakeup_device_power_package(acpi_handle handle,
 +                                      struct acpi_device_wakeup *wakeup)
  {
        struct acpi_buffer buffer = { ACPI_ALLOCATE_BUFFER, NULL };
        union acpi_object *package = NULL;
        union acpi_object *element = NULL;
        acpi_status status;
 -      int i = 0;
 +      int err = -ENODATA;
  
        if (!wakeup)
 -              return AE_BAD_PARAMETER;
 +              return -EINVAL;
 +
 +      INIT_LIST_HEAD(&wakeup->resources);
  
        /* _PRW */
        status = acpi_evaluate_object(handle, "_PRW", NULL, &buffer);
        if (ACPI_FAILURE(status)) {
                ACPI_EXCEPTION((AE_INFO, status, "Evaluating _PRW"));
 -              return status;
 +              return err;
        }
  
        package = (union acpi_object *)buffer.pointer;
  
 -      if (!package || (package->package.count < 2)) {
 -              status = AE_BAD_DATA;
 +      if (!package || package->package.count < 2)
                goto out;
 -      }
  
        element = &(package->package.elements[0]);
 -      if (!element) {
 -              status = AE_BAD_DATA;
 +      if (!element)
                goto out;
 -      }
 +
        if (element->type == ACPI_TYPE_PACKAGE) {
                if ((element->package.count < 2) ||
                    (element->package.elements[0].type !=
                     ACPI_TYPE_LOCAL_REFERENCE)
 -                  || (element->package.elements[1].type != ACPI_TYPE_INTEGER)) {
 -                      status = AE_BAD_DATA;
 +                  || (element->package.elements[1].type != ACPI_TYPE_INTEGER))
                        goto out;
 -              }
 +
                wakeup->gpe_device =
                    element->package.elements[0].reference.handle;
                wakeup->gpe_number =
                wakeup->gpe_device = NULL;
                wakeup->gpe_number = element->integer.value;
        } else {
 -              status = AE_BAD_DATA;
                goto out;
        }
  
        element = &(package->package.elements[1]);
 -      if (element->type != ACPI_TYPE_INTEGER) {
 -              status = AE_BAD_DATA;
 +      if (element->type != ACPI_TYPE_INTEGER)
                goto out;
 -      }
 +
        wakeup->sleep_state = element->integer.value;
  
 -      if ((package->package.count - 2) > ACPI_MAX_HANDLES) {
 -              status = AE_NO_MEMORY;
 +      err = acpi_extract_power_resources(package, 2, &wakeup->resources);
 +      if (err)
                goto out;
 -      }
 -      wakeup->resources.count = package->package.count - 2;
 -      for (i = 0; i < wakeup->resources.count; i++) {
 -              element = &(package->package.elements[i + 2]);
 -              if (element->type != ACPI_TYPE_LOCAL_REFERENCE) {
 -                      status = AE_BAD_DATA;
 -                      goto out;
 -              }
  
 -              wakeup->resources.handles[i] = element->reference.handle;
 -      }
 +      if (!list_empty(&wakeup->resources)) {
 +              int sleep_state;
  
 +              sleep_state = acpi_power_min_system_level(&wakeup->resources);
 +              if (sleep_state < wakeup->sleep_state) {
 +                      acpi_handle_warn(handle, "Overriding _PRW sleep state "
 +                                       "(S%d) by S%d from power resources\n",
 +                                       (int)wakeup->sleep_state, sleep_state);
 +                      wakeup->sleep_state = sleep_state;
 +              }
 +      }
        acpi_setup_gpe_for_wake(handle, wakeup->gpe_device, wakeup->gpe_number);
  
   out:
        kfree(buffer.pointer);
 -
 -      return status;
 +      return err;
  }
  
  static void acpi_bus_set_run_wake_flags(struct acpi_device *device)
@@@ -1052,17 -975,17 +1052,17 @@@ static void acpi_bus_get_wakeup_device_
  {
        acpi_handle temp;
        acpi_status status = 0;
 -      int psw_error;
 +      int err;
  
        /* Presence of _PRW indicates wake capable */
        status = acpi_get_handle(device->handle, "_PRW", &temp);
        if (ACPI_FAILURE(status))
                return;
  
 -      status = acpi_bus_extract_wakeup_device_power_package(device->handle,
 -                                                            &device->wakeup);
 -      if (ACPI_FAILURE(status)) {
 -              ACPI_EXCEPTION((AE_INFO, status, "Extracting _PRW package"));
 +      err = acpi_bus_extract_wakeup_device_power_package(device->handle,
 +                                                         &device->wakeup);
 +      if (err) {
 +              dev_err(&device->dev, "_PRW evaluation error: %d\n", err);
                return;
        }
  
         * So it is necessary to call _DSW object first. Only when it is not
         * present will the _PSW object used.
         */
 -      psw_error = acpi_device_sleep_wake(device, 0, 0, 0);
 -      if (psw_error)
 +      err = acpi_device_sleep_wake(device, 0, 0, 0);
 +      if (err)
                ACPI_DEBUG_PRINT((ACPI_DB_INFO,
                                "error in _DSW or _PSW evaluation\n"));
  }
  
 -static void acpi_bus_add_power_resource(acpi_handle handle);
 +static void acpi_bus_init_power_state(struct acpi_device *device, int state)
 +{
 +      struct acpi_device_power_state *ps = &device->power.states[state];
 +      char pathname[5] = { '_', 'P', 'R', '0' + state, '\0' };
 +      struct acpi_buffer buffer = { ACPI_ALLOCATE_BUFFER, NULL };
 +      acpi_handle handle;
 +      acpi_status status;
 +
 +      INIT_LIST_HEAD(&ps->resources);
 +
 +      /* Evaluate "_PRx" to get referenced power resources */
 +      status = acpi_evaluate_object(device->handle, pathname, NULL, &buffer);
 +      if (ACPI_SUCCESS(status)) {
 +              union acpi_object *package = buffer.pointer;
 +
 +              if (buffer.length && package
 +                  && package->type == ACPI_TYPE_PACKAGE
 +                  && package->package.count) {
 +                      int err = acpi_extract_power_resources(package, 0,
 +                                                             &ps->resources);
 +                      if (!err)
 +                              device->power.flags.power_resources = 1;
 +              }
 +              ACPI_FREE(buffer.pointer);
 +      }
 +
 +      /* Evaluate "_PSx" to see if we can do explicit sets */
 +      pathname[2] = 'S';
 +      status = acpi_get_handle(device->handle, pathname, &handle);
 +      if (ACPI_SUCCESS(status))
 +              ps->flags.explicit_set = 1;
  
 -static int acpi_bus_get_power_flags(struct acpi_device *device)
 +      /*
 +       * State is valid if there are means to put the device into it.
 +       * D3hot is only valid if _PR3 present.
 +       */
 +      if (!list_empty(&ps->resources)
 +          || (ps->flags.explicit_set && state < ACPI_STATE_D3_HOT)) {
 +              ps->flags.valid = 1;
 +              ps->flags.os_accessible = 1;
 +      }
 +
 +      ps->power = -1;         /* Unknown - driver assigned */
 +      ps->latency = -1;       /* Unknown - driver assigned */
 +}
 +
 +static void acpi_bus_get_power_flags(struct acpi_device *device)
  {
 -      acpi_status status = 0;
 -      acpi_handle handle = NULL;
 -      u32 i = 0;
 +      acpi_status status;
 +      acpi_handle handle;
 +      u32 i;
  
 +      /* Presence of _PS0|_PR0 indicates 'power manageable' */
 +      status = acpi_get_handle(device->handle, "_PS0", &handle);
 +      if (ACPI_FAILURE(status)) {
 +              status = acpi_get_handle(device->handle, "_PR0", &handle);
 +              if (ACPI_FAILURE(status))
 +                      return;
 +      }
 +
 +      device->flags.power_manageable = 1;
  
        /*
         * Power Management Flags
        /*
         * Enumerate supported power management states
         */
 -      for (i = ACPI_STATE_D0; i <= ACPI_STATE_D3_HOT; i++) {
 -              struct acpi_device_power_state *ps = &device->power.states[i];
 -              char object_name[5] = { '_', 'P', 'R', '0' + i, '\0' };
 +      for (i = ACPI_STATE_D0; i <= ACPI_STATE_D3_HOT; i++)
 +              acpi_bus_init_power_state(device, i);
  
 -              /* Evaluate "_PRx" to se if power resources are referenced */
 -              acpi_evaluate_reference(device->handle, object_name, NULL,
 -                                      &ps->resources);
 -              if (ps->resources.count) {
 -                      int j;
 -
 -                      device->power.flags.power_resources = 1;
 -                      for (j = 0; j < ps->resources.count; j++)
 -                              acpi_bus_add_power_resource(ps->resources.handles[j]);
 -              }
 -
 -              /* Evaluate "_PSx" to see if we can do explicit sets */
 -              object_name[2] = 'S';
 -              status = acpi_get_handle(device->handle, object_name, &handle);
 -              if (ACPI_SUCCESS(status))
 -                      ps->flags.explicit_set = 1;
 -
 -              /*
 -               * State is valid if there are means to put the device into it.
 -               * D3hot is only valid if _PR3 present.
 -               */
 -              if (ps->resources.count ||
 -                  (ps->flags.explicit_set && i < ACPI_STATE_D3_HOT)) {
 -                      ps->flags.valid = 1;
 -                      ps->flags.os_accessible = 1;
 -              }
 -
 -              ps->power = -1; /* Unknown - driver assigned */
 -              ps->latency = -1;       /* Unknown - driver assigned */
 -      }
 +      INIT_LIST_HEAD(&device->power.states[ACPI_STATE_D3_COLD].resources);
  
        /* Set defaults for D0 and D3 states (always valid) */
        device->power.states[ACPI_STATE_D0].flags.valid = 1;
                        device->power.flags.power_resources)
                device->power.states[ACPI_STATE_D3_COLD].flags.os_accessible = 1;
  
 -      acpi_bus_init_power(device);
 -
 -      return 0;
 +      if (acpi_bus_init_power(device)) {
 +              acpi_free_power_resources_lists(device);
 +              device->flags.power_manageable = 0;
 +      }
  }
  
 -static int acpi_bus_get_flags(struct acpi_device *device)
 +static void acpi_bus_get_flags(struct acpi_device *device)
  {
        acpi_status status = AE_OK;
        acpi_handle temp = NULL;
  
 -
        /* Presence of _STA indicates 'dynamic_status' */
        status = acpi_get_handle(device->handle, "_STA", &temp);
        if (ACPI_SUCCESS(status))
                if (ACPI_SUCCESS(status))
                        device->flags.ejectable = 1;
        }
 -
 -      /* Power resources cannot be power manageable. */
 -      if (device->device_type == ACPI_BUS_TYPE_POWER)
 -              return 0;
 -
 -      /* Presence of _PS0|_PR0 indicates 'power manageable' */
 -      status = acpi_get_handle(device->handle, "_PS0", &temp);
 -      if (ACPI_FAILURE(status))
 -              status = acpi_get_handle(device->handle, "_PR0", &temp);
 -      if (ACPI_SUCCESS(status))
 -              device->flags.power_manageable = 1;
 -
 -      /* TBD: Performance management */
 -
 -      return 0;
  }
  
  static void acpi_device_get_busid(struct acpi_device *device)
@@@ -1430,25 -1345,27 +1430,25 @@@ static void acpi_device_set_id(struct a
        }
  }
  
 -static int acpi_device_set_context(struct acpi_device *device)
 +void acpi_init_device_object(struct acpi_device *device, acpi_handle handle,
 +                           int type, unsigned long long sta)
  {
 -      acpi_status status;
 -
 -      /*
 -       * Context
 -       * -------
 -       * Attach this 'struct acpi_device' to the ACPI object.  This makes
 -       * resolutions from handle->device very efficient.  Fixed hardware
 -       * devices have no handles, so we skip them.
 -       */
 -      if (!device->handle)
 -              return 0;
 -
 -      status = acpi_attach_data(device->handle,
 -                                acpi_bus_data_handler, device);
 -      if (ACPI_SUCCESS(status))
 -              return 0;
 +      INIT_LIST_HEAD(&device->pnp.ids);
 +      device->device_type = type;
 +      device->handle = handle;
 +      device->parent = acpi_bus_get_parent(handle);
 +      STRUCT_TO_INT(device->status) = sta;
 +      acpi_device_get_busid(device);
 +      acpi_device_set_id(device);
 +      acpi_bus_get_flags(device);
 +      device_initialize(&device->dev);
 +      dev_set_uevent_suppress(&device->dev, true);
 +}
  
 -      printk(KERN_ERR PREFIX "Error attaching device data\n");
 -      return -ENODEV;
 +void acpi_device_add_finalize(struct acpi_device *device)
 +{
 +      dev_set_uevent_suppress(&device->dev, false);
 +      kobject_uevent(&device->dev.kobj, KOBJ_ADD);
  }
  
  static int acpi_add_single_object(struct acpi_device **child,
                return -ENOMEM;
        }
  
 -      INIT_LIST_HEAD(&device->pnp.ids);
 -      device->device_type = type;
 -      device->handle = handle;
 -      device->parent = acpi_bus_get_parent(handle);
 -      STRUCT_TO_INT(device->status) = sta;
 -
 -      acpi_device_get_busid(device);
 -
 -      /*
 -       * Flags
 -       * -----
 -       * Note that we only look for object handles -- cannot evaluate objects
 -       * until we know the device is present and properly initialized.
 -       */
 -      result = acpi_bus_get_flags(device);
 -      if (result)
 -              goto end;
 -
 -      /*
 -       * Initialize Device
 -       * -----------------
 -       * TBD: Synch with Core's enumeration/initialization process.
 -       */
 -      acpi_device_set_id(device);
 -
 -      /*
 -       * Power Management
 -       * ----------------
 -       */
 -      if (device->flags.power_manageable) {
 -              result = acpi_bus_get_power_flags(device);
 -              if (result)
 -                      goto end;
 -      }
 -
 -      /*
 -       * Wakeup device management
 -       *-----------------------
 -       */
 +      acpi_init_device_object(device, handle, type, sta);
 +      acpi_bus_get_power_flags(device);
        acpi_bus_get_wakeup_device_flags(device);
  
 -      /*
 -       * Performance Management
 -       * ----------------------
 -       */
 -      if (device->flags.performance_manageable) {
 -              result = acpi_bus_get_perf_flags(device);
 -              if (result)
 -                      goto end;
 -      }
 -
 -      if ((result = acpi_device_set_context(device)))
 -              goto end;
 -
        device->flags.match_driver = match_driver;
 -      result = acpi_device_register(device);
 -
 -end:
 -      if (!result) {
 -              acpi_get_name(handle, ACPI_FULL_PATHNAME, &buffer);
 -              ACPI_DEBUG_PRINT((ACPI_DB_INFO,
 -                      "Adding %s [%s] parent %s\n", dev_name(&device->dev),
 -                       (char *) buffer.pointer,
 -                       device->parent ? dev_name(&device->parent->dev) :
 -                                        "(null)"));
 -              kfree(buffer.pointer);
 -              *child = device;
 -      } else
 +      result = acpi_device_add(device, acpi_device_release);
 +      if (result) {
                acpi_device_release(&device->dev);
 +              return result;
 +      }
  
 -      return result;
 -}
 -
 -#define ACPI_STA_DEFAULT (ACPI_STA_DEVICE_PRESENT | ACPI_STA_DEVICE_ENABLED | \
 -                        ACPI_STA_DEVICE_UI      | ACPI_STA_DEVICE_FUNCTIONING)
 -
 -static void acpi_bus_add_power_resource(acpi_handle handle)
 -{
 -      struct acpi_device *device = NULL;
 -
 -      acpi_bus_get_device(handle, &device);
 -      if (!device)
 -              acpi_add_single_object(&device, handle, ACPI_BUS_TYPE_POWER,
 -                                      ACPI_STA_DEFAULT, true);
 +      acpi_power_add_remove_device(device, true);
 +      acpi_device_add_finalize(device);
 +      acpi_get_name(handle, ACPI_FULL_PATHNAME, &buffer);
 +      ACPI_DEBUG_PRINT((ACPI_DB_INFO, "Added %s [%s] parent %s\n",
 +              dev_name(&device->dev), (char *) buffer.pointer,
 +              device->parent ? dev_name(&device->parent->dev) : "(null)"));
 +      kfree(buffer.pointer);
 +      *child = device;
 +      return 0;
  }
  
  static int acpi_bus_type_and_status(acpi_handle handle, int *type,
@@@ -1543,26 -1524,20 +1543,26 @@@ static acpi_status acpi_bus_check_add(a
        if (result)
                return AE_OK;
  
 +      if (type == ACPI_BUS_TYPE_POWER) {
 +              acpi_add_power_resource(handle);
 +              return AE_OK;
 +      }
 +
        if (!(sta & ACPI_STA_DEVICE_PRESENT) &&
            !(sta & ACPI_STA_DEVICE_FUNCTIONING)) {
                struct acpi_device_wakeup wakeup;
                acpi_handle temp;
  
                status = acpi_get_handle(handle, "_PRW", &temp);
 -              if (ACPI_SUCCESS(status))
 +              if (ACPI_SUCCESS(status)) {
                        acpi_bus_extract_wakeup_device_power_package(handle,
                                                                     &wakeup);
 +                      acpi_power_resources_list_free(&wakeup.resources);
 +              }
                return AE_CTRL_DEPTH;
        }
  
 -      acpi_add_single_object(&device, handle, type, sta,
 -                             type == ACPI_BUS_TYPE_POWER);
 +      acpi_add_single_object(&device, handle, type, sta, false);
        if (!device)
                return AE_CTRL_DEPTH;
  
  static acpi_status acpi_bus_device_attach(acpi_handle handle, u32 lvl_not_used,
                                          void *not_used, void **ret_not_used)
  {
+       const struct acpi_device_id *id;
        acpi_status status = AE_OK;
        struct acpi_device *device;
        unsigned long long sta_not_used;
        if (acpi_bus_get_device(handle, &device))
                return AE_CTRL_DEPTH;
  
-       if (!acpi_match_device_ids(device, acpi_platform_device_ids)) {
+       id = __acpi_match_device(device, acpi_platform_device_ids);
+       if (id) {
                /* This is a known good platform device. */
-               acpi_create_platform_device(device);
+               acpi_create_platform_device(device, id->driver_data);
        } else if (device_attach(&device->dev) < 0) {
                status = AE_CTRL_DEPTH;
        }
        return status;
  }
  
 -static int acpi_bus_scan(acpi_handle handle)
 -{
 -      void *device = NULL;
 -
 -      if (ACPI_SUCCESS(acpi_bus_check_add(handle, 0, NULL, &device)))
 -              acpi_walk_namespace(ACPI_TYPE_ANY, handle, ACPI_UINT32_MAX,
 -                                  acpi_bus_check_add, NULL, NULL, &device);
 -
 -      if (!device)
 -              return -ENODEV;
 -
 -      if (ACPI_SUCCESS(acpi_bus_device_attach(handle, 0, NULL, NULL)))
 -              acpi_walk_namespace(ACPI_TYPE_ANY, handle, ACPI_UINT32_MAX,
 -                                  acpi_bus_device_attach, NULL, NULL, NULL);
 -
 -      return 0;
 -}
 -
  /**
 - * acpi_bus_add - Add ACPI device node objects in a given namespace scope.
 + * acpi_bus_scan - Add ACPI device node objects in a given namespace scope.
   * @handle: Root of the namespace scope to scan.
   *
   * Scan a given ACPI tree (probably recently hot-plugged) and create and add
   * in the table trunk from which the kernel could create a device and add an
   * appropriate driver.
   */
 -int acpi_bus_add(acpi_handle handle)
 +int acpi_bus_scan(acpi_handle handle)
  {
 -      int err;
 +      void *device = NULL;
 +      int error = 0;
  
 -      err = acpi_bus_scan(handle);
 -      if (err)
 -              return err;
 +      mutex_lock(&acpi_scan_lock);
  
 -      acpi_update_all_gpes();
 -      return 0;
 +      if (ACPI_SUCCESS(acpi_bus_check_add(handle, 0, NULL, &device)))
 +              acpi_walk_namespace(ACPI_TYPE_ANY, handle, ACPI_UINT32_MAX,
 +                                  acpi_bus_check_add, NULL, NULL, &device);
 +
 +      if (!device)
 +              error = -ENODEV;
 +      else if (ACPI_SUCCESS(acpi_bus_device_attach(handle, 0, NULL, NULL)))
 +              acpi_walk_namespace(ACPI_TYPE_ANY, handle, ACPI_UINT32_MAX,
 +                                  acpi_bus_device_attach, NULL, NULL, NULL);
 +
 +      mutex_unlock(&acpi_scan_lock);
 +      return error;
  }
 -EXPORT_SYMBOL(acpi_bus_add);
 +EXPORT_SYMBOL(acpi_bus_scan);
  
  static acpi_status acpi_bus_device_detach(acpi_handle handle, u32 lvl_not_used,
                                          void *not_used, void **ret_not_used)
@@@ -1659,10 -1645,8 +1661,10 @@@ static acpi_status acpi_bus_remove(acpi
        return AE_OK;
  }
  
 -int acpi_bus_trim(struct acpi_device *start)
 +void acpi_bus_trim(struct acpi_device *start)
  {
 +      mutex_lock(&acpi_scan_lock);
 +
        /*
         * Execute acpi_bus_device_detach() as a post-order callback to detach
         * all ACPI drivers from the device nodes being removed.
        acpi_walk_namespace(ACPI_TYPE_ANY, start->handle, ACPI_UINT32_MAX, NULL,
                            acpi_bus_remove, NULL, NULL);
        acpi_bus_remove(start->handle, 0, NULL, NULL);
 -      return 0;
 +
 +      mutex_unlock(&acpi_scan_lock);
  }
  EXPORT_SYMBOL_GPL(acpi_bus_trim);
  
@@@ -1716,7 -1699,9 +1718,8 @@@ int __init acpi_scan_init(void
                printk(KERN_ERR PREFIX "Could not register bus type\n");
        }
  
 -      acpi_power_init();
        acpi_pci_root_init();
+       acpi_csrt_init();
  
        /*
         * Enumerate devices in the ACPI namespace.
                return result;
  
        result = acpi_bus_get_device(ACPI_ROOT_OBJECT, &acpi_root);
 -      if (!result)
 -              result = acpi_bus_scan_fixed();
 -
        if (result)
 +              return result;
 +
 +      result = acpi_bus_scan_fixed();
 +      if (result) {
                acpi_device_unregister(acpi_root);
 -      else
 -              acpi_update_all_gpes();
 +              return result;
 +      }
  
 -      return result;
 +      acpi_update_all_gpes();
 +      return 0;
  }