2019-05-27 09:55:15 +03:00
// SPDX-License-Identifier: GPL-2.0-only
2011-01-13 04:00:22 +03:00
/*
* Copyright ( C ) 2010 OKI SEMICONDUCTOR Co . , LTD .
*/
2011-07-03 21:38:09 +04:00
# include <linux/module.h>
2011-01-13 04:00:22 +03:00
# include <linux/kernel.h>
2011-05-25 04:13:43 +04:00
# include <linux/slab.h>
2011-01-13 04:00:22 +03:00
# include <linux/pci.h>
2018-04-13 16:17:11 +03:00
# include <linux/gpio/driver.h>
2011-08-05 08:04:21 +04:00
# include <linux/interrupt.h>
# include <linux/irq.h>
# define IOH_EDGE_FALLING 0
# define IOH_EDGE_RISING BIT(0)
# define IOH_LEVEL_L BIT(1)
# define IOH_LEVEL_H (BIT(0) | BIT(1))
# define IOH_EDGE_BOTH BIT(2)
# define IOH_IM_MASK (BIT(0) | BIT(1) | BIT(2))
# define IOH_IRQ_BASE 0
2011-01-13 04:00:22 +03:00
struct ioh_reg_comn {
u32 ien ;
u32 istatus ;
u32 idisp ;
u32 iclr ;
u32 imask ;
u32 imaskclr ;
u32 po ;
u32 pi ;
u32 pm ;
u32 im_0 ;
u32 im_1 ;
u32 reserved ;
} ;
struct ioh_regs {
struct ioh_reg_comn regs [ 8 ] ;
u32 reserve1 [ 16 ] ;
u32 ioh_sel_reg [ 4 ] ;
u32 reserve2 [ 11 ] ;
u32 srst ;
} ;
/**
* struct ioh_gpio_reg_data - The register store data .
2020-06-30 16:33:38 +03:00
* @ ien_reg : To store contents of interrupt enable register .
2011-08-05 08:04:21 +04:00
* @ imask_reg : To store contents of interrupt mask regist
2011-01-13 04:00:22 +03:00
* @ po_reg : To store contents of PO register .
* @ pm_reg : To store contents of PM register .
2011-08-05 08:04:21 +04:00
* @ im0_reg : To store contents of interrupt mode regist0
* @ im1_reg : To store contents of interrupt mode regist1
2011-08-05 08:04:22 +04:00
* @ use_sel_reg : To store contents of GPIO_USE_SEL0 ~ 3
2011-01-13 04:00:22 +03:00
*/
struct ioh_gpio_reg_data {
2011-08-05 08:04:21 +04:00
u32 ien_reg ;
u32 imask_reg ;
2011-01-13 04:00:22 +03:00
u32 po_reg ;
u32 pm_reg ;
2011-08-05 08:04:21 +04:00
u32 im0_reg ;
u32 im1_reg ;
2011-08-05 08:04:22 +04:00
u32 use_sel_reg ;
2011-01-13 04:00:22 +03:00
} ;
/**
* struct ioh_gpio - GPIO private data structure .
* @ base : PCI base address of Memory mapped I / O register .
* @ reg : Memory mapped IOH GPIO register list .
* @ dev : Pointer to device structure .
* @ gpio : Data for GPIO infrastructure .
* @ ioh_gpio_reg : Memory mapped Register data is saved here
* when suspend .
2011-08-05 08:04:22 +04:00
* @ gpio_use_sel : Save GPIO_USE_SEL1 ~ 4 register for PM
2011-01-13 04:00:22 +03:00
* @ ch : Indicate GPIO channel
2011-08-05 08:04:21 +04:00
* @ irq_base : Save base of IRQ number for interrupt
2012-07-29 06:54:42 +04:00
* @ spinlock : Used for register access protection
2011-01-13 04:00:22 +03:00
*/
struct ioh_gpio {
void __iomem * base ;
struct ioh_regs __iomem * reg ;
struct device * dev ;
struct gpio_chip gpio ;
struct ioh_gpio_reg_data ioh_gpio_reg ;
2011-08-05 08:04:22 +04:00
u32 gpio_use_sel ;
2011-01-13 04:00:22 +03:00
int ch ;
2011-08-05 08:04:21 +04:00
int irq_base ;
spinlock_t spinlock ;
2011-01-13 04:00:22 +03:00
} ;
static const int num_ports [ ] = { 6 , 12 , 16 , 16 , 15 , 16 , 16 , 12 } ;
static void ioh_gpio_set ( struct gpio_chip * gpio , unsigned nr , int val )
{
u32 reg_val ;
2015-12-07 12:12:05 +03:00
struct ioh_gpio * chip = gpiochip_get_data ( gpio ) ;
2012-07-29 06:54:42 +04:00
unsigned long flags ;
2011-01-13 04:00:22 +03:00
2012-07-29 06:54:42 +04:00
spin_lock_irqsave ( & chip - > spinlock , flags ) ;
2011-01-13 04:00:22 +03:00
reg_val = ioread32 ( & chip - > reg - > regs [ chip - > ch ] . po ) ;
if ( val )
reg_val | = ( 1 < < nr ) ;
else
reg_val & = ~ ( 1 < < nr ) ;
iowrite32 ( reg_val , & chip - > reg - > regs [ chip - > ch ] . po ) ;
2012-07-29 06:54:42 +04:00
spin_unlock_irqrestore ( & chip - > spinlock , flags ) ;
2011-01-13 04:00:22 +03:00
}
static int ioh_gpio_get ( struct gpio_chip * gpio , unsigned nr )
{
2015-12-07 12:12:05 +03:00
struct ioh_gpio * chip = gpiochip_get_data ( gpio ) ;
2011-01-13 04:00:22 +03:00
2015-12-21 13:18:44 +03:00
return ! ! ( ioread32 ( & chip - > reg - > regs [ chip - > ch ] . pi ) & ( 1 < < nr ) ) ;
2011-01-13 04:00:22 +03:00
}
static int ioh_gpio_direction_output ( struct gpio_chip * gpio , unsigned nr ,
int val )
{
2015-12-07 12:12:05 +03:00
struct ioh_gpio * chip = gpiochip_get_data ( gpio ) ;
2011-01-13 04:00:22 +03:00
u32 pm ;
u32 reg_val ;
2012-07-29 06:54:42 +04:00
unsigned long flags ;
2011-01-13 04:00:22 +03:00
2012-07-29 06:54:42 +04:00
spin_lock_irqsave ( & chip - > spinlock , flags ) ;
2011-01-13 04:00:22 +03:00
pm = ioread32 ( & chip - > reg - > regs [ chip - > ch ] . pm ) &
( ( 1 < < num_ports [ chip - > ch ] ) - 1 ) ;
pm | = ( 1 < < nr ) ;
iowrite32 ( pm , & chip - > reg - > regs [ chip - > ch ] . pm ) ;
reg_val = ioread32 ( & chip - > reg - > regs [ chip - > ch ] . po ) ;
if ( val )
reg_val | = ( 1 < < nr ) ;
else
reg_val & = ~ ( 1 < < nr ) ;
2011-03-25 02:17:14 +03:00
iowrite32 ( reg_val , & chip - > reg - > regs [ chip - > ch ] . po ) ;
2011-01-13 04:00:22 +03:00
2012-07-29 06:54:42 +04:00
spin_unlock_irqrestore ( & chip - > spinlock , flags ) ;
2011-01-13 04:00:22 +03:00
return 0 ;
}
static int ioh_gpio_direction_input ( struct gpio_chip * gpio , unsigned nr )
{
2015-12-07 12:12:05 +03:00
struct ioh_gpio * chip = gpiochip_get_data ( gpio ) ;
2011-01-13 04:00:22 +03:00
u32 pm ;
2012-07-29 06:54:42 +04:00
unsigned long flags ;
2011-01-13 04:00:22 +03:00
2012-07-29 06:54:42 +04:00
spin_lock_irqsave ( & chip - > spinlock , flags ) ;
2011-01-13 04:00:22 +03:00
pm = ioread32 ( & chip - > reg - > regs [ chip - > ch ] . pm ) &
( ( 1 < < num_ports [ chip - > ch ] ) - 1 ) ;
pm & = ~ ( 1 < < nr ) ;
iowrite32 ( pm , & chip - > reg - > regs [ chip - > ch ] . pm ) ;
2012-07-29 06:54:42 +04:00
spin_unlock_irqrestore ( & chip - > spinlock , flags ) ;
2011-01-13 04:00:22 +03:00
return 0 ;
}
2011-05-25 04:13:43 +04:00
# ifdef CONFIG_PM
2011-01-13 04:00:22 +03:00
/*
* Save register configuration and disable interrupts .
*/
static void ioh_gpio_save_reg_conf ( struct ioh_gpio * chip )
{
2011-08-05 08:04:22 +04:00
int i ;
for ( i = 0 ; i < 8 ; i + + , chip + + ) {
chip - > ioh_gpio_reg . po_reg =
ioread32 ( & chip - > reg - > regs [ chip - > ch ] . po ) ;
chip - > ioh_gpio_reg . pm_reg =
ioread32 ( & chip - > reg - > regs [ chip - > ch ] . pm ) ;
chip - > ioh_gpio_reg . ien_reg =
ioread32 ( & chip - > reg - > regs [ chip - > ch ] . ien ) ;
chip - > ioh_gpio_reg . imask_reg =
ioread32 ( & chip - > reg - > regs [ chip - > ch ] . imask ) ;
chip - > ioh_gpio_reg . im0_reg =
ioread32 ( & chip - > reg - > regs [ chip - > ch ] . im_0 ) ;
chip - > ioh_gpio_reg . im1_reg =
ioread32 ( & chip - > reg - > regs [ chip - > ch ] . im_1 ) ;
if ( i < 4 )
chip - > ioh_gpio_reg . use_sel_reg =
ioread32 ( & chip - > reg - > ioh_sel_reg [ i ] ) ;
}
2011-01-13 04:00:22 +03:00
}
/*
* This function restores the register configuration of the GPIO device .
*/
static void ioh_gpio_restore_reg_conf ( struct ioh_gpio * chip )
{
2011-08-05 08:04:22 +04:00
int i ;
for ( i = 0 ; i < 8 ; i + + , chip + + ) {
iowrite32 ( chip - > ioh_gpio_reg . po_reg ,
& chip - > reg - > regs [ chip - > ch ] . po ) ;
iowrite32 ( chip - > ioh_gpio_reg . pm_reg ,
& chip - > reg - > regs [ chip - > ch ] . pm ) ;
iowrite32 ( chip - > ioh_gpio_reg . ien_reg ,
& chip - > reg - > regs [ chip - > ch ] . ien ) ;
iowrite32 ( chip - > ioh_gpio_reg . imask_reg ,
& chip - > reg - > regs [ chip - > ch ] . imask ) ;
iowrite32 ( chip - > ioh_gpio_reg . im0_reg ,
& chip - > reg - > regs [ chip - > ch ] . im_0 ) ;
iowrite32 ( chip - > ioh_gpio_reg . im1_reg ,
& chip - > reg - > regs [ chip - > ch ] . im_1 ) ;
if ( i < 4 )
iowrite32 ( chip - > ioh_gpio_reg . use_sel_reg ,
& chip - > reg - > ioh_sel_reg [ i ] ) ;
}
2011-01-13 04:00:22 +03:00
}
2011-05-25 04:13:43 +04:00
# endif
2011-01-13 04:00:22 +03:00
2011-08-05 08:04:21 +04:00
static int ioh_gpio_to_irq ( struct gpio_chip * gpio , unsigned offset )
{
2015-12-07 12:12:05 +03:00
struct ioh_gpio * chip = gpiochip_get_data ( gpio ) ;
2011-08-05 08:04:21 +04:00
return chip - > irq_base + offset ;
}
2011-01-13 04:00:22 +03:00
static void ioh_gpio_setup ( struct ioh_gpio * chip , int num_port )
{
struct gpio_chip * gpio = & chip - > gpio ;
gpio - > label = dev_name ( chip - > dev ) ;
gpio - > owner = THIS_MODULE ;
gpio - > direction_input = ioh_gpio_direction_input ;
gpio - > get = ioh_gpio_get ;
gpio - > direction_output = ioh_gpio_direction_output ;
gpio - > set = ioh_gpio_set ;
gpio - > dbg_show = NULL ;
gpio - > base = - 1 ;
gpio - > ngpio = num_port ;
2013-12-04 17:42:46 +04:00
gpio - > can_sleep = false ;
2011-08-05 08:04:21 +04:00
gpio - > to_irq = ioh_gpio_to_irq ;
}
static int ioh_irq_type ( struct irq_data * d , unsigned int type )
{
u32 im ;
2012-01-15 13:57:34 +04:00
void __iomem * im_reg ;
2011-08-05 08:04:21 +04:00
u32 ien ;
u32 im_pos ;
int ch ;
unsigned long flags ;
u32 val ;
int irq = d - > irq ;
struct irq_chip_generic * gc = irq_data_get_irq_chip_data ( d ) ;
struct ioh_gpio * chip = gc - > private ;
ch = irq - chip - > irq_base ;
if ( irq < = chip - > irq_base + 7 ) {
im_reg = & chip - > reg - > regs [ chip - > ch ] . im_0 ;
im_pos = ch ;
} else {
im_reg = & chip - > reg - > regs [ chip - > ch ] . im_1 ;
im_pos = ch - 8 ;
}
dev_dbg ( chip - > dev , " %s:irq=%d type=%d ch=%d pos=%d type=%d \n " ,
__func__ , irq , type , ch , im_pos , type ) ;
spin_lock_irqsave ( & chip - > spinlock , flags ) ;
switch ( type ) {
case IRQ_TYPE_EDGE_RISING :
val = IOH_EDGE_RISING ;
break ;
case IRQ_TYPE_EDGE_FALLING :
val = IOH_EDGE_FALLING ;
break ;
case IRQ_TYPE_EDGE_BOTH :
val = IOH_EDGE_BOTH ;
break ;
case IRQ_TYPE_LEVEL_HIGH :
val = IOH_LEVEL_H ;
break ;
case IRQ_TYPE_LEVEL_LOW :
val = IOH_LEVEL_L ;
break ;
case IRQ_TYPE_PROBE :
goto end ;
default :
dev_warn ( chip - > dev , " %s: unknown type(%dd) " ,
__func__ , type ) ;
goto end ;
}
/* Set interrupt mode */
im = ioread32 ( im_reg ) & ~ ( IOH_IM_MASK < < ( im_pos * 4 ) ) ;
iowrite32 ( im | ( val < < ( im_pos * 4 ) ) , im_reg ) ;
/* iclr */
iowrite32 ( BIT ( ch ) , & chip - > reg - > regs [ chip - > ch ] . iclr ) ;
/* IMASKCLR */
iowrite32 ( BIT ( ch ) , & chip - > reg - > regs [ chip - > ch ] . imaskclr ) ;
/* Enable interrupt */
ien = ioread32 ( & chip - > reg - > regs [ chip - > ch ] . ien ) ;
iowrite32 ( ien | BIT ( ch ) , & chip - > reg - > regs [ chip - > ch ] . ien ) ;
end :
spin_unlock_irqrestore ( & chip - > spinlock , flags ) ;
return 0 ;
}
static void ioh_irq_unmask ( struct irq_data * d )
{
struct irq_chip_generic * gc = irq_data_get_irq_chip_data ( d ) ;
struct ioh_gpio * chip = gc - > private ;
iowrite32 ( 1 < < ( d - > irq - chip - > irq_base ) ,
& chip - > reg - > regs [ chip - > ch ] . imaskclr ) ;
}
static void ioh_irq_mask ( struct irq_data * d )
{
struct irq_chip_generic * gc = irq_data_get_irq_chip_data ( d ) ;
struct ioh_gpio * chip = gc - > private ;
iowrite32 ( 1 < < ( d - > irq - chip - > irq_base ) ,
& chip - > reg - > regs [ chip - > ch ] . imask ) ;
}
2011-12-13 19:53:50 +04:00
static void ioh_irq_disable ( struct irq_data * d )
{
struct irq_chip_generic * gc = irq_data_get_irq_chip_data ( d ) ;
struct ioh_gpio * chip = gc - > private ;
unsigned long flags ;
u32 ien ;
spin_lock_irqsave ( & chip - > spinlock , flags ) ;
ien = ioread32 ( & chip - > reg - > regs [ chip - > ch ] . ien ) ;
ien & = ~ ( 1 < < ( d - > irq - chip - > irq_base ) ) ;
iowrite32 ( ien , & chip - > reg - > regs [ chip - > ch ] . ien ) ;
spin_unlock_irqrestore ( & chip - > spinlock , flags ) ;
}
static void ioh_irq_enable ( struct irq_data * d )
{
struct irq_chip_generic * gc = irq_data_get_irq_chip_data ( d ) ;
struct ioh_gpio * chip = gc - > private ;
unsigned long flags ;
u32 ien ;
spin_lock_irqsave ( & chip - > spinlock , flags ) ;
ien = ioread32 ( & chip - > reg - > regs [ chip - > ch ] . ien ) ;
ien | = 1 < < ( d - > irq - chip - > irq_base ) ;
iowrite32 ( ien , & chip - > reg - > regs [ chip - > ch ] . ien ) ;
spin_unlock_irqrestore ( & chip - > spinlock , flags ) ;
}
2011-08-05 08:04:21 +04:00
static irqreturn_t ioh_gpio_handler ( int irq , void * dev_id )
{
struct ioh_gpio * chip = dev_id ;
u32 reg_val ;
int i , j ;
int ret = IRQ_NONE ;
2011-12-13 19:53:49 +04:00
for ( i = 0 ; i < 8 ; i + + , chip + + ) {
2011-08-05 08:04:21 +04:00
reg_val = ioread32 ( & chip - > reg - > regs [ i ] . istatus ) ;
for ( j = 0 ; j < num_ports [ i ] ; j + + ) {
if ( reg_val & BIT ( j ) ) {
dev_dbg ( chip - > dev ,
" %s:[%d]:irq=%d status=0x%x \n " ,
__func__ , j , irq , reg_val ) ;
iowrite32 ( BIT ( j ) ,
& chip - > reg - > regs [ chip - > ch ] . iclr ) ;
generic_handle_irq ( chip - > irq_base + j ) ;
ret = IRQ_HANDLED ;
}
}
}
return ret ;
}
2017-05-25 11:37:38 +03:00
static int ioh_gpio_alloc_generic_chip ( struct ioh_gpio * chip ,
unsigned int irq_start ,
unsigned int num )
2011-08-05 08:04:21 +04:00
{
struct irq_chip_generic * gc ;
struct irq_chip_type * ct ;
2017-08-09 15:25:04 +03:00
int rv ;
2011-08-05 08:04:21 +04:00
2017-08-09 15:25:04 +03:00
gc = devm_irq_alloc_generic_chip ( chip - > dev , " ioh_gpio " , 1 , irq_start ,
chip - > base , handle_simple_irq ) ;
2017-05-25 11:37:38 +03:00
if ( ! gc )
return - ENOMEM ;
2011-08-05 08:04:21 +04:00
gc - > private = chip ;
ct = gc - > chip_types ;
ct - > chip . irq_mask = ioh_irq_mask ;
ct - > chip . irq_unmask = ioh_irq_unmask ;
ct - > chip . irq_set_type = ioh_irq_type ;
2011-12-13 19:53:50 +04:00
ct - > chip . irq_disable = ioh_irq_disable ;
ct - > chip . irq_enable = ioh_irq_enable ;
2011-08-05 08:04:21 +04:00
2017-08-09 15:25:04 +03:00
rv = devm_irq_setup_generic_chip ( chip - > dev , gc , IRQ_MSK ( num ) ,
IRQ_GC_INIT_MASK_CACHE ,
IRQ_NOREQUEST | IRQ_NOPROBE , 0 ) ;
2017-05-25 11:37:38 +03:00
2017-08-09 15:25:04 +03:00
return rv ;
2011-01-13 04:00:22 +03:00
}
2012-11-19 22:22:34 +04:00
static int ioh_gpio_probe ( struct pci_dev * pdev ,
2011-01-13 04:00:22 +03:00
const struct pci_device_id * id )
{
int ret ;
2011-08-05 08:04:21 +04:00
int i , j ;
2011-01-13 04:00:22 +03:00
struct ioh_gpio * chip ;
void __iomem * base ;
2012-01-15 13:57:34 +04:00
void * chip_save ;
2011-08-05 08:04:21 +04:00
int irq_base ;
2011-01-13 04:00:22 +03:00
ret = pci_enable_device ( pdev ) ;
if ( ret ) {
dev_err ( & pdev - > dev , " %s : pci_enable_device failed " , __func__ ) ;
goto err_pci_enable ;
}
ret = pci_request_regions ( pdev , KBUILD_MODNAME ) ;
if ( ret ) {
dev_err ( & pdev - > dev , " pci_request_regions failed-%d " , ret ) ;
goto err_request_regions ;
}
base = pci_iomap ( pdev , 1 , 0 ) ;
2012-01-15 13:57:43 +04:00
if ( ! base ) {
2011-01-13 04:00:22 +03:00
dev_err ( & pdev - > dev , " %s : pci_iomap failed " , __func__ ) ;
ret = - ENOMEM ;
goto err_iomap ;
}
treewide: kzalloc() -> kcalloc()
The kzalloc() function has a 2-factor argument form, kcalloc(). This
patch replaces cases of:
kzalloc(a * b, gfp)
with:
kcalloc(a * b, gfp)
as well as handling cases of:
kzalloc(a * b * c, gfp)
with:
kzalloc(array3_size(a, b, c), gfp)
as it's slightly less ugly than:
kzalloc_array(array_size(a, b), c, gfp)
This does, however, attempt to ignore constant size factors like:
kzalloc(4 * 1024, gfp)
though any constants defined via macros get caught up in the conversion.
Any factors with a sizeof() of "unsigned char", "char", and "u8" were
dropped, since they're redundant.
The Coccinelle script used for this was:
// Fix redundant parens around sizeof().
@@
type TYPE;
expression THING, E;
@@
(
kzalloc(
- (sizeof(TYPE)) * E
+ sizeof(TYPE) * E
, ...)
|
kzalloc(
- (sizeof(THING)) * E
+ sizeof(THING) * E
, ...)
)
// Drop single-byte sizes and redundant parens.
@@
expression COUNT;
typedef u8;
typedef __u8;
@@
(
kzalloc(
- sizeof(u8) * (COUNT)
+ COUNT
, ...)
|
kzalloc(
- sizeof(__u8) * (COUNT)
+ COUNT
, ...)
|
kzalloc(
- sizeof(char) * (COUNT)
+ COUNT
, ...)
|
kzalloc(
- sizeof(unsigned char) * (COUNT)
+ COUNT
, ...)
|
kzalloc(
- sizeof(u8) * COUNT
+ COUNT
, ...)
|
kzalloc(
- sizeof(__u8) * COUNT
+ COUNT
, ...)
|
kzalloc(
- sizeof(char) * COUNT
+ COUNT
, ...)
|
kzalloc(
- sizeof(unsigned char) * COUNT
+ COUNT
, ...)
)
// 2-factor product with sizeof(type/expression) and identifier or constant.
@@
type TYPE;
expression THING;
identifier COUNT_ID;
constant COUNT_CONST;
@@
(
- kzalloc
+ kcalloc
(
- sizeof(TYPE) * (COUNT_ID)
+ COUNT_ID, sizeof(TYPE)
, ...)
|
- kzalloc
+ kcalloc
(
- sizeof(TYPE) * COUNT_ID
+ COUNT_ID, sizeof(TYPE)
, ...)
|
- kzalloc
+ kcalloc
(
- sizeof(TYPE) * (COUNT_CONST)
+ COUNT_CONST, sizeof(TYPE)
, ...)
|
- kzalloc
+ kcalloc
(
- sizeof(TYPE) * COUNT_CONST
+ COUNT_CONST, sizeof(TYPE)
, ...)
|
- kzalloc
+ kcalloc
(
- sizeof(THING) * (COUNT_ID)
+ COUNT_ID, sizeof(THING)
, ...)
|
- kzalloc
+ kcalloc
(
- sizeof(THING) * COUNT_ID
+ COUNT_ID, sizeof(THING)
, ...)
|
- kzalloc
+ kcalloc
(
- sizeof(THING) * (COUNT_CONST)
+ COUNT_CONST, sizeof(THING)
, ...)
|
- kzalloc
+ kcalloc
(
- sizeof(THING) * COUNT_CONST
+ COUNT_CONST, sizeof(THING)
, ...)
)
// 2-factor product, only identifiers.
@@
identifier SIZE, COUNT;
@@
- kzalloc
+ kcalloc
(
- SIZE * COUNT
+ COUNT, SIZE
, ...)
// 3-factor product with 1 sizeof(type) or sizeof(expression), with
// redundant parens removed.
@@
expression THING;
identifier STRIDE, COUNT;
type TYPE;
@@
(
kzalloc(
- sizeof(TYPE) * (COUNT) * (STRIDE)
+ array3_size(COUNT, STRIDE, sizeof(TYPE))
, ...)
|
kzalloc(
- sizeof(TYPE) * (COUNT) * STRIDE
+ array3_size(COUNT, STRIDE, sizeof(TYPE))
, ...)
|
kzalloc(
- sizeof(TYPE) * COUNT * (STRIDE)
+ array3_size(COUNT, STRIDE, sizeof(TYPE))
, ...)
|
kzalloc(
- sizeof(TYPE) * COUNT * STRIDE
+ array3_size(COUNT, STRIDE, sizeof(TYPE))
, ...)
|
kzalloc(
- sizeof(THING) * (COUNT) * (STRIDE)
+ array3_size(COUNT, STRIDE, sizeof(THING))
, ...)
|
kzalloc(
- sizeof(THING) * (COUNT) * STRIDE
+ array3_size(COUNT, STRIDE, sizeof(THING))
, ...)
|
kzalloc(
- sizeof(THING) * COUNT * (STRIDE)
+ array3_size(COUNT, STRIDE, sizeof(THING))
, ...)
|
kzalloc(
- sizeof(THING) * COUNT * STRIDE
+ array3_size(COUNT, STRIDE, sizeof(THING))
, ...)
)
// 3-factor product with 2 sizeof(variable), with redundant parens removed.
@@
expression THING1, THING2;
identifier COUNT;
type TYPE1, TYPE2;
@@
(
kzalloc(
- sizeof(TYPE1) * sizeof(TYPE2) * COUNT
+ array3_size(COUNT, sizeof(TYPE1), sizeof(TYPE2))
, ...)
|
kzalloc(
- sizeof(TYPE1) * sizeof(THING2) * (COUNT)
+ array3_size(COUNT, sizeof(TYPE1), sizeof(TYPE2))
, ...)
|
kzalloc(
- sizeof(THING1) * sizeof(THING2) * COUNT
+ array3_size(COUNT, sizeof(THING1), sizeof(THING2))
, ...)
|
kzalloc(
- sizeof(THING1) * sizeof(THING2) * (COUNT)
+ array3_size(COUNT, sizeof(THING1), sizeof(THING2))
, ...)
|
kzalloc(
- sizeof(TYPE1) * sizeof(THING2) * COUNT
+ array3_size(COUNT, sizeof(TYPE1), sizeof(THING2))
, ...)
|
kzalloc(
- sizeof(TYPE1) * sizeof(THING2) * (COUNT)
+ array3_size(COUNT, sizeof(TYPE1), sizeof(THING2))
, ...)
)
// 3-factor product, only identifiers, with redundant parens removed.
@@
identifier STRIDE, SIZE, COUNT;
@@
(
kzalloc(
- (COUNT) * STRIDE * SIZE
+ array3_size(COUNT, STRIDE, SIZE)
, ...)
|
kzalloc(
- COUNT * (STRIDE) * SIZE
+ array3_size(COUNT, STRIDE, SIZE)
, ...)
|
kzalloc(
- COUNT * STRIDE * (SIZE)
+ array3_size(COUNT, STRIDE, SIZE)
, ...)
|
kzalloc(
- (COUNT) * (STRIDE) * SIZE
+ array3_size(COUNT, STRIDE, SIZE)
, ...)
|
kzalloc(
- COUNT * (STRIDE) * (SIZE)
+ array3_size(COUNT, STRIDE, SIZE)
, ...)
|
kzalloc(
- (COUNT) * STRIDE * (SIZE)
+ array3_size(COUNT, STRIDE, SIZE)
, ...)
|
kzalloc(
- (COUNT) * (STRIDE) * (SIZE)
+ array3_size(COUNT, STRIDE, SIZE)
, ...)
|
kzalloc(
- COUNT * STRIDE * SIZE
+ array3_size(COUNT, STRIDE, SIZE)
, ...)
)
// Any remaining multi-factor products, first at least 3-factor products,
// when they're not all constants...
@@
expression E1, E2, E3;
constant C1, C2, C3;
@@
(
kzalloc(C1 * C2 * C3, ...)
|
kzalloc(
- (E1) * E2 * E3
+ array3_size(E1, E2, E3)
, ...)
|
kzalloc(
- (E1) * (E2) * E3
+ array3_size(E1, E2, E3)
, ...)
|
kzalloc(
- (E1) * (E2) * (E3)
+ array3_size(E1, E2, E3)
, ...)
|
kzalloc(
- E1 * E2 * E3
+ array3_size(E1, E2, E3)
, ...)
)
// And then all remaining 2 factors products when they're not all constants,
// keeping sizeof() as the second factor argument.
@@
expression THING, E1, E2;
type TYPE;
constant C1, C2, C3;
@@
(
kzalloc(sizeof(THING) * C2, ...)
|
kzalloc(sizeof(TYPE) * C2, ...)
|
kzalloc(C1 * C2 * C3, ...)
|
kzalloc(C1 * C2, ...)
|
- kzalloc
+ kcalloc
(
- sizeof(TYPE) * (E2)
+ E2, sizeof(TYPE)
, ...)
|
- kzalloc
+ kcalloc
(
- sizeof(TYPE) * E2
+ E2, sizeof(TYPE)
, ...)
|
- kzalloc
+ kcalloc
(
- sizeof(THING) * (E2)
+ E2, sizeof(THING)
, ...)
|
- kzalloc
+ kcalloc
(
- sizeof(THING) * E2
+ E2, sizeof(THING)
, ...)
|
- kzalloc
+ kcalloc
(
- (E1) * E2
+ E1, E2
, ...)
|
- kzalloc
+ kcalloc
(
- (E1) * (E2)
+ E1, E2
, ...)
|
- kzalloc
+ kcalloc
(
- E1 * E2
+ E1, E2
, ...)
)
Signed-off-by: Kees Cook <keescook@chromium.org>
2018-06-13 00:03:40 +03:00
chip_save = kcalloc ( 8 , sizeof ( * chip ) , GFP_KERNEL ) ;
2011-01-13 04:00:22 +03:00
if ( chip_save = = NULL ) {
ret = - ENOMEM ;
goto err_kzalloc ;
}
chip = chip_save ;
for ( i = 0 ; i < 8 ; i + + , chip + + ) {
chip - > dev = & pdev - > dev ;
chip - > base = base ;
chip - > reg = chip - > base ;
chip - > ch = i ;
2012-02-01 06:50:05 +04:00
spin_lock_init ( & chip - > spinlock ) ;
2011-01-13 04:00:22 +03:00
ioh_gpio_setup ( chip , num_ports [ i ] ) ;
2015-12-07 12:12:05 +03:00
ret = gpiochip_add_data ( & chip - > gpio , chip ) ;
2011-01-13 04:00:22 +03:00
if ( ret ) {
dev_err ( & pdev - > dev , " IOH gpio: Failed to register GPIO \n " ) ;
goto err_gpiochip_add ;
}
}
chip = chip_save ;
2011-08-05 08:04:21 +04:00
for ( j = 0 ; j < 8 ; j + + , chip + + ) {
2017-03-04 19:23:33 +03:00
irq_base = devm_irq_alloc_descs ( & pdev - > dev , - 1 , IOH_IRQ_BASE ,
num_ports [ j ] , NUMA_NO_NODE ) ;
2011-08-05 08:04:21 +04:00
if ( irq_base < 0 ) {
dev_warn ( & pdev - > dev ,
" ml_ioh_gpio: Failed to get IRQ base num \n " ) ;
2013-05-21 19:11:10 +04:00
ret = irq_base ;
2017-03-04 19:23:33 +03:00
goto err_gpiochip_add ;
2011-08-05 08:04:21 +04:00
}
chip - > irq_base = irq_base ;
2017-05-25 11:37:38 +03:00
ret = ioh_gpio_alloc_generic_chip ( chip ,
irq_base , num_ports [ j ] ) ;
if ( ret )
goto err_gpiochip_add ;
2011-08-05 08:04:21 +04:00
}
chip = chip_save ;
2017-03-04 19:23:33 +03:00
ret = devm_request_irq ( & pdev - > dev , pdev - > irq , ioh_gpio_handler ,
IRQF_SHARED , KBUILD_MODNAME , chip ) ;
2011-08-05 08:04:21 +04:00
if ( ret ! = 0 ) {
dev_err ( & pdev - > dev ,
" %s request_irq failed \n " , __func__ ) ;
2017-03-04 19:23:33 +03:00
goto err_gpiochip_add ;
2011-08-05 08:04:21 +04:00
}
2011-01-13 04:00:22 +03:00
pci_set_drvdata ( pdev , chip ) ;
return 0 ;
err_gpiochip_add :
2018-07-23 19:53:30 +03:00
chip = chip_save ;
2011-06-14 15:12:57 +04:00
while ( - - i > = 0 ) {
2014-07-13 00:30:12 +04:00
gpiochip_remove ( & chip - > gpio ) ;
2018-07-23 19:53:30 +03:00
chip + + ;
2011-01-13 04:00:22 +03:00
}
kfree ( chip_save ) ;
err_kzalloc :
pci_iounmap ( pdev , base ) ;
err_iomap :
pci_release_regions ( pdev ) ;
err_request_regions :
pci_disable_device ( pdev ) ;
err_pci_enable :
dev_err ( & pdev - > dev , " %s Failed returns %d \n " , __func__ , ret ) ;
return ret ;
}
2012-11-19 22:25:50 +04:00
static void ioh_gpio_remove ( struct pci_dev * pdev )
2011-01-13 04:00:22 +03:00
{
int i ;
struct ioh_gpio * chip = pci_get_drvdata ( pdev ) ;
2012-01-15 13:57:34 +04:00
void * chip_save ;
2011-01-13 04:00:22 +03:00
chip_save = chip ;
2011-08-05 08:04:21 +04:00
2017-03-04 19:23:33 +03:00
for ( i = 0 ; i < 8 ; i + + , chip + + )
2014-07-13 00:30:12 +04:00
gpiochip_remove ( & chip - > gpio ) ;
2011-01-13 04:00:22 +03:00
chip = chip_save ;
pci_iounmap ( pdev , chip - > base ) ;
pci_release_regions ( pdev ) ;
pci_disable_device ( pdev ) ;
kfree ( chip ) ;
}
# ifdef CONFIG_PM
static int ioh_gpio_suspend ( struct pci_dev * pdev , pm_message_t state )
{
s32 ret ;
struct ioh_gpio * chip = pci_get_drvdata ( pdev ) ;
2011-08-05 08:04:22 +04:00
unsigned long flags ;
2011-01-13 04:00:22 +03:00
2011-08-05 08:04:22 +04:00
spin_lock_irqsave ( & chip - > spinlock , flags ) ;
2011-01-13 04:00:22 +03:00
ioh_gpio_save_reg_conf ( chip ) ;
2011-08-05 08:04:22 +04:00
spin_unlock_irqrestore ( & chip - > spinlock , flags ) ;
2011-01-13 04:00:22 +03:00
ret = pci_save_state ( pdev ) ;
if ( ret ) {
dev_err ( & pdev - > dev , " pci_save_state Failed-%d \n " , ret ) ;
return ret ;
}
pci_disable_device ( pdev ) ;
pci_set_power_state ( pdev , PCI_D0 ) ;
ret = pci_enable_wake ( pdev , PCI_D0 , 1 ) ;
if ( ret )
dev_err ( & pdev - > dev , " pci_enable_wake Failed -%d \n " , ret ) ;
return 0 ;
}
static int ioh_gpio_resume ( struct pci_dev * pdev )
{
s32 ret ;
struct ioh_gpio * chip = pci_get_drvdata ( pdev ) ;
2011-08-05 08:04:22 +04:00
unsigned long flags ;
2011-01-13 04:00:22 +03:00
ret = pci_enable_wake ( pdev , PCI_D0 , 0 ) ;
pci_set_power_state ( pdev , PCI_D0 ) ;
ret = pci_enable_device ( pdev ) ;
if ( ret ) {
dev_err ( & pdev - > dev , " pci_enable_device Failed-%d " , ret ) ;
return ret ;
}
pci_restore_state ( pdev ) ;
2011-08-05 08:04:22 +04:00
spin_lock_irqsave ( & chip - > spinlock , flags ) ;
2011-01-13 04:00:22 +03:00
iowrite32 ( 0x01 , & chip - > reg - > srst ) ;
iowrite32 ( 0x00 , & chip - > reg - > srst ) ;
ioh_gpio_restore_reg_conf ( chip ) ;
2011-08-05 08:04:22 +04:00
spin_unlock_irqrestore ( & chip - > spinlock , flags ) ;
2011-01-13 04:00:22 +03:00
return 0 ;
}
# else
# define ioh_gpio_suspend NULL
# define ioh_gpio_resume NULL
# endif
2013-12-03 03:08:45 +04:00
static const struct pci_device_id ioh_gpio_pcidev_id [ ] = {
2011-01-13 04:00:22 +03:00
{ PCI_DEVICE ( PCI_VENDOR_ID_ROHM , 0x802E ) } ,
{ 0 , }
} ;
2011-03-12 01:58:30 +03:00
MODULE_DEVICE_TABLE ( pci , ioh_gpio_pcidev_id ) ;
2011-01-13 04:00:22 +03:00
static struct pci_driver ioh_gpio_driver = {
. name = " ml_ioh_gpio " ,
. id_table = ioh_gpio_pcidev_id ,
. probe = ioh_gpio_probe ,
2012-11-19 22:20:08 +04:00
. remove = ioh_gpio_remove ,
2011-01-13 04:00:22 +03:00
. suspend = ioh_gpio_suspend ,
. resume = ioh_gpio_resume
} ;
2012-04-06 16:13:30 +04:00
module_pci_driver ( ioh_gpio_driver ) ;
2011-01-13 04:00:22 +03:00
MODULE_DESCRIPTION ( " OKI SEMICONDUCTOR ML-IOH series GPIO Driver " ) ;
MODULE_LICENSE ( " GPL " ) ;