2019-06-04 11:11:33 +03:00
// SPDX-License-Identifier: GPL-2.0-only
2014-08-19 20:17:35 +04:00
/*
* Intel Low Power Subsystem PWM controller driver
*
* Copyright ( C ) 2014 , Intel Corporation
*
* Derived from the original pwm - lpss . c
*/
# include <linux/acpi.h>
# include <linux/kernel.h>
# include <linux/module.h>
# include <linux/platform_device.h>
2015-10-26 13:58:27 +03:00
# include <linux/pm_runtime.h>
2014-08-19 20:17:35 +04:00
# include "pwm-lpss.h"
2017-01-28 18:10:43 +03:00
2014-08-19 20:17:35 +04:00
static int pwm_lpss_probe_platform ( struct platform_device * pdev )
{
const struct pwm_lpss_boardinfo * info ;
const struct acpi_device_id * id ;
struct pwm_lpss_chip * lpwm ;
2022-09-27 19:24:17 +03:00
void __iomem * base ;
2014-08-19 20:17:35 +04:00
id = acpi_match_device ( pdev - > dev . driver - > acpi_match_table , & pdev - > dev ) ;
if ( ! id )
return - ENODEV ;
info = ( const struct pwm_lpss_boardinfo * ) id - > driver_data ;
2022-09-27 19:24:17 +03:00
base = devm_platform_ioremap_resource ( pdev , 0 ) ;
if ( IS_ERR ( base ) )
return PTR_ERR ( base ) ;
lpwm = pwm_lpss_probe ( & pdev - > dev , base , info ) ;
2014-08-19 20:17:35 +04:00
if ( IS_ERR ( lpwm ) )
return PTR_ERR ( lpwm ) ;
platform_set_drvdata ( pdev , lpwm ) ;
2015-10-26 13:58:27 +03:00
2020-11-09 13:57:25 +03:00
/*
* On Cherry Trail devices the GFX0 . _PS0 AML checks if the controller
* is on and if it is not on it turns it on and restores what it
* believes is the correct state to the PWM controller .
* Because of this we must disallow direct - complete , which keeps the
* controller ( runtime ) suspended on resume , to avoid 2 issues :
* 1. The controller getting turned on without the linux - pm code
* knowing about this . On devices where the controller is unused
* this causes it to stay on during the next suspend causing high
* battery drain ( because S0i3 is not reached )
* 2. The state restoring code unexpectedly messing with the controller
2020-11-09 13:57:26 +03:00
*
* Leaving the controller runtime - suspended ( skipping runtime - resume +
* normal - suspend ) during suspend is fine .
2020-11-09 13:57:25 +03:00
*/
if ( info - > other_devices_aml_touches_pwm_regs )
2020-11-09 13:57:26 +03:00
dev_pm_set_driver_flags ( & pdev - > dev , DPM_FLAG_NO_DIRECT_COMPLETE |
DPM_FLAG_SMART_SUSPEND ) ;
2020-11-09 13:57:25 +03:00
2015-10-26 13:58:27 +03:00
pm_runtime_set_active ( & pdev - > dev ) ;
pm_runtime_enable ( & pdev - > dev ) ;
2014-08-19 20:17:35 +04:00
return 0 ;
}
static int pwm_lpss_remove_platform ( struct platform_device * pdev )
{
2015-10-26 13:58:27 +03:00
pm_runtime_disable ( & pdev - > dev ) ;
2021-04-07 11:01:55 +03:00
return 0 ;
2014-08-19 20:17:35 +04:00
}
static const struct acpi_device_id pwm_lpss_acpi_match [ ] = {
{ " 80860F09 " , ( unsigned long ) & pwm_lpss_byt_info } ,
{ " 80862288 " , ( unsigned long ) & pwm_lpss_bsw_info } ,
2018-10-12 13:12:25 +03:00
{ " 80862289 " , ( unsigned long ) & pwm_lpss_bsw_info } ,
2015-10-20 16:53:07 +03:00
{ " 80865AC8 " , ( unsigned long ) & pwm_lpss_bxt_info } ,
2014-08-19 20:17:35 +04:00
{ } ,
} ;
MODULE_DEVICE_TABLE ( acpi , pwm_lpss_acpi_match ) ;
static struct platform_driver pwm_lpss_driver_platform = {
. driver = {
. name = " pwm-lpss " ,
. acpi_match_table = pwm_lpss_acpi_match ,
} ,
. probe = pwm_lpss_probe_platform ,
. remove = pwm_lpss_remove_platform ,
} ;
module_platform_driver ( pwm_lpss_driver_platform ) ;
MODULE_DESCRIPTION ( " PWM platform driver for Intel LPSS " ) ;
MODULE_LICENSE ( " GPL v2 " ) ;
2022-09-27 19:24:16 +03:00
MODULE_IMPORT_NS ( PWM_LPSS ) ;
2014-08-19 20:17:35 +04:00
MODULE_ALIAS ( " platform:pwm-lpss " ) ;