Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
L
linux
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
Analytics
Analytics
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Commits
Issue Boards
Open sidebar
Kirill Smelkov
linux
Commits
5eff79fe
Commit
5eff79fe
authored
Oct 28, 2013
by
Mark Brown
Browse files
Options
Browse Files
Download
Plain Diff
Merge remote-tracking branch 'regmap/topic/async' into regmap-next
parents
70c1c86d
04c50ccf
Changes
5
Hide whitespace changes
Inline
Side-by-side
Showing
5 changed files
with
225 additions
and
50 deletions
+225
-50
drivers/base/regmap/internal.h
drivers/base/regmap/internal.h
+3
-2
drivers/base/regmap/regcache.c
drivers/base/regmap/regcache.c
+15
-4
drivers/base/regmap/regmap-spi.c
drivers/base/regmap/regmap-spi.c
+2
-1
drivers/base/regmap/regmap.c
drivers/base/regmap/regmap.c
+174
-43
include/linux/regmap.h
include/linux/regmap.h
+31
-0
No files found.
drivers/base/regmap/internal.h
View file @
5eff79fe
...
@@ -44,7 +44,6 @@ struct regmap_format {
...
@@ -44,7 +44,6 @@ struct regmap_format {
struct
regmap_async
{
struct
regmap_async
{
struct
list_head
list
;
struct
list_head
list
;
struct
work_struct
cleanup
;
struct
regmap
*
map
;
struct
regmap
*
map
;
void
*
work_buf
;
void
*
work_buf
;
};
};
...
@@ -64,9 +63,11 @@ struct regmap {
...
@@ -64,9 +63,11 @@ struct regmap {
void
*
bus_context
;
void
*
bus_context
;
const
char
*
name
;
const
char
*
name
;
bool
async
;
spinlock_t
async_lock
;
spinlock_t
async_lock
;
wait_queue_head_t
async_waitq
;
wait_queue_head_t
async_waitq
;
struct
list_head
async_list
;
struct
list_head
async_list
;
struct
list_head
async_free
;
int
async_ret
;
int
async_ret
;
#ifdef CONFIG_DEBUG_FS
#ifdef CONFIG_DEBUG_FS
...
@@ -218,7 +219,7 @@ bool regcache_set_val(struct regmap *map, void *base, unsigned int idx,
...
@@ -218,7 +219,7 @@ bool regcache_set_val(struct regmap *map, void *base, unsigned int idx,
int
regcache_lookup_reg
(
struct
regmap
*
map
,
unsigned
int
reg
);
int
regcache_lookup_reg
(
struct
regmap
*
map
,
unsigned
int
reg
);
int
_regmap_raw_write
(
struct
regmap
*
map
,
unsigned
int
reg
,
int
_regmap_raw_write
(
struct
regmap
*
map
,
unsigned
int
reg
,
const
void
*
val
,
size_t
val_len
,
bool
async
);
const
void
*
val
,
size_t
val_len
);
void
regmap_async_complete_cb
(
struct
regmap_async
*
async
,
int
ret
);
void
regmap_async_complete_cb
(
struct
regmap_async
*
async
,
int
ret
);
...
...
drivers/base/regmap/regcache.c
View file @
5eff79fe
...
@@ -307,6 +307,8 @@ int regcache_sync(struct regmap *map)
...
@@ -307,6 +307,8 @@ int regcache_sync(struct regmap *map)
if
(
!
map
->
cache_dirty
)
if
(
!
map
->
cache_dirty
)
goto
out
;
goto
out
;
map
->
async
=
true
;
/* Apply any patch first */
/* Apply any patch first */
map
->
cache_bypass
=
1
;
map
->
cache_bypass
=
1
;
for
(
i
=
0
;
i
<
map
->
patch_regs
;
i
++
)
{
for
(
i
=
0
;
i
<
map
->
patch_regs
;
i
++
)
{
...
@@ -332,11 +334,15 @@ int regcache_sync(struct regmap *map)
...
@@ -332,11 +334,15 @@ int regcache_sync(struct regmap *map)
map
->
cache_dirty
=
false
;
map
->
cache_dirty
=
false
;
out:
out:
trace_regcache_sync
(
map
->
dev
,
name
,
"stop"
);
/* Restore the bypass state */
/* Restore the bypass state */
map
->
async
=
false
;
map
->
cache_bypass
=
bypass
;
map
->
cache_bypass
=
bypass
;
map
->
unlock
(
map
->
lock_arg
);
map
->
unlock
(
map
->
lock_arg
);
regmap_async_complete
(
map
);
trace_regcache_sync
(
map
->
dev
,
name
,
"stop"
);
return
ret
;
return
ret
;
}
}
EXPORT_SYMBOL_GPL
(
regcache_sync
);
EXPORT_SYMBOL_GPL
(
regcache_sync
);
...
@@ -375,17 +381,23 @@ int regcache_sync_region(struct regmap *map, unsigned int min,
...
@@ -375,17 +381,23 @@ int regcache_sync_region(struct regmap *map, unsigned int min,
if
(
!
map
->
cache_dirty
)
if
(
!
map
->
cache_dirty
)
goto
out
;
goto
out
;
map
->
async
=
true
;
if
(
map
->
cache_ops
->
sync
)
if
(
map
->
cache_ops
->
sync
)
ret
=
map
->
cache_ops
->
sync
(
map
,
min
,
max
);
ret
=
map
->
cache_ops
->
sync
(
map
,
min
,
max
);
else
else
ret
=
regcache_default_sync
(
map
,
min
,
max
);
ret
=
regcache_default_sync
(
map
,
min
,
max
);
out:
out:
trace_regcache_sync
(
map
->
dev
,
name
,
"stop region"
);
/* Restore the bypass state */
/* Restore the bypass state */
map
->
cache_bypass
=
bypass
;
map
->
cache_bypass
=
bypass
;
map
->
async
=
false
;
map
->
unlock
(
map
->
lock_arg
);
map
->
unlock
(
map
->
lock_arg
);
regmap_async_complete
(
map
);
trace_regcache_sync
(
map
->
dev
,
name
,
"stop region"
);
return
ret
;
return
ret
;
}
}
EXPORT_SYMBOL_GPL
(
regcache_sync_region
);
EXPORT_SYMBOL_GPL
(
regcache_sync_region
);
...
@@ -631,8 +643,7 @@ static int regcache_sync_block_raw_flush(struct regmap *map, const void **data,
...
@@ -631,8 +643,7 @@ static int regcache_sync_block_raw_flush(struct regmap *map, const void **data,
map
->
cache_bypass
=
1
;
map
->
cache_bypass
=
1
;
ret
=
_regmap_raw_write
(
map
,
base
,
*
data
,
count
*
val_bytes
,
ret
=
_regmap_raw_write
(
map
,
base
,
*
data
,
count
*
val_bytes
);
false
);
map
->
cache_bypass
=
0
;
map
->
cache_bypass
=
0
;
...
...
drivers/base/regmap/regmap-spi.c
View file @
5eff79fe
...
@@ -73,7 +73,8 @@ static int regmap_spi_async_write(void *context,
...
@@ -73,7 +73,8 @@ static int regmap_spi_async_write(void *context,
spi_message_init
(
&
async
->
m
);
spi_message_init
(
&
async
->
m
);
spi_message_add_tail
(
&
async
->
t
[
0
],
&
async
->
m
);
spi_message_add_tail
(
&
async
->
t
[
0
],
&
async
->
m
);
spi_message_add_tail
(
&
async
->
t
[
1
],
&
async
->
m
);
if
(
val
)
spi_message_add_tail
(
&
async
->
t
[
1
],
&
async
->
m
);
async
->
m
.
complete
=
regmap_spi_complete
;
async
->
m
.
complete
=
regmap_spi_complete
;
async
->
m
.
context
=
async
;
async
->
m
.
context
=
async
;
...
...
drivers/base/regmap/regmap.c
View file @
5eff79fe
...
@@ -42,15 +42,6 @@ static int _regmap_bus_formatted_write(void *context, unsigned int reg,
...
@@ -42,15 +42,6 @@ static int _regmap_bus_formatted_write(void *context, unsigned int reg,
static
int
_regmap_bus_raw_write
(
void
*
context
,
unsigned
int
reg
,
static
int
_regmap_bus_raw_write
(
void
*
context
,
unsigned
int
reg
,
unsigned
int
val
);
unsigned
int
val
);
static
void
async_cleanup
(
struct
work_struct
*
work
)
{
struct
regmap_async
*
async
=
container_of
(
work
,
struct
regmap_async
,
cleanup
);
kfree
(
async
->
work_buf
);
kfree
(
async
);
}
bool
regmap_reg_in_ranges
(
unsigned
int
reg
,
bool
regmap_reg_in_ranges
(
unsigned
int
reg
,
const
struct
regmap_range
*
ranges
,
const
struct
regmap_range
*
ranges
,
unsigned
int
nranges
)
unsigned
int
nranges
)
...
@@ -465,6 +456,7 @@ struct regmap *regmap_init(struct device *dev,
...
@@ -465,6 +456,7 @@ struct regmap *regmap_init(struct device *dev,
spin_lock_init
(
&
map
->
async_lock
);
spin_lock_init
(
&
map
->
async_lock
);
INIT_LIST_HEAD
(
&
map
->
async_list
);
INIT_LIST_HEAD
(
&
map
->
async_list
);
INIT_LIST_HEAD
(
&
map
->
async_free
);
init_waitqueue_head
(
&
map
->
async_waitq
);
init_waitqueue_head
(
&
map
->
async_waitq
);
if
(
config
->
read_flag_mask
||
config
->
write_flag_mask
)
{
if
(
config
->
read_flag_mask
||
config
->
write_flag_mask
)
{
...
@@ -942,12 +934,22 @@ EXPORT_SYMBOL_GPL(regmap_reinit_cache);
...
@@ -942,12 +934,22 @@ EXPORT_SYMBOL_GPL(regmap_reinit_cache);
*/
*/
void
regmap_exit
(
struct
regmap
*
map
)
void
regmap_exit
(
struct
regmap
*
map
)
{
{
struct
regmap_async
*
async
;
regcache_exit
(
map
);
regcache_exit
(
map
);
regmap_debugfs_exit
(
map
);
regmap_debugfs_exit
(
map
);
regmap_range_exit
(
map
);
regmap_range_exit
(
map
);
if
(
map
->
bus
&&
map
->
bus
->
free_context
)
if
(
map
->
bus
&&
map
->
bus
->
free_context
)
map
->
bus
->
free_context
(
map
->
bus_context
);
map
->
bus
->
free_context
(
map
->
bus_context
);
kfree
(
map
->
work_buf
);
kfree
(
map
->
work_buf
);
while
(
!
list_empty
(
&
map
->
async_free
))
{
async
=
list_first_entry_or_null
(
&
map
->
async_free
,
struct
regmap_async
,
list
);
list_del
(
&
async
->
list
);
kfree
(
async
->
work_buf
);
kfree
(
async
);
}
kfree
(
map
);
kfree
(
map
);
}
}
EXPORT_SYMBOL_GPL
(
regmap_exit
);
EXPORT_SYMBOL_GPL
(
regmap_exit
);
...
@@ -1039,7 +1041,7 @@ static int _regmap_select_page(struct regmap *map, unsigned int *reg,
...
@@ -1039,7 +1041,7 @@ static int _regmap_select_page(struct regmap *map, unsigned int *reg,
}
}
int
_regmap_raw_write
(
struct
regmap
*
map
,
unsigned
int
reg
,
int
_regmap_raw_write
(
struct
regmap
*
map
,
unsigned
int
reg
,
const
void
*
val
,
size_t
val_len
,
bool
async
)
const
void
*
val
,
size_t
val_len
)
{
{
struct
regmap_range_node
*
range
;
struct
regmap_range_node
*
range
;
unsigned
long
flags
;
unsigned
long
flags
;
...
@@ -1091,7 +1093,7 @@ int _regmap_raw_write(struct regmap *map, unsigned int reg,
...
@@ -1091,7 +1093,7 @@ int _regmap_raw_write(struct regmap *map, unsigned int reg,
dev_dbg
(
map
->
dev
,
"Writing window %d/%zu
\n
"
,
dev_dbg
(
map
->
dev
,
"Writing window %d/%zu
\n
"
,
win_residue
,
val_len
/
map
->
format
.
val_bytes
);
win_residue
,
val_len
/
map
->
format
.
val_bytes
);
ret
=
_regmap_raw_write
(
map
,
reg
,
val
,
win_residue
*
ret
=
_regmap_raw_write
(
map
,
reg
,
val
,
win_residue
*
map
->
format
.
val_bytes
,
async
);
map
->
format
.
val_bytes
);
if
(
ret
!=
0
)
if
(
ret
!=
0
)
return
ret
;
return
ret
;
...
@@ -1114,49 +1116,72 @@ int _regmap_raw_write(struct regmap *map, unsigned int reg,
...
@@ -1114,49 +1116,72 @@ int _regmap_raw_write(struct regmap *map, unsigned int reg,
u8
[
0
]
|=
map
->
write_flag_mask
;
u8
[
0
]
|=
map
->
write_flag_mask
;
if
(
async
&&
map
->
bus
->
async_write
)
{
/*
struct
regmap_async
*
async
=
map
->
bus
->
async_alloc
();
* Essentially all I/O mechanisms will be faster with a single
if
(
!
async
)
* buffer to write. Since register syncs often generate raw
return
-
ENOMEM
;
* writes of single registers optimise that case.
*/
if
(
val
!=
work_val
&&
val_len
==
map
->
format
.
val_bytes
)
{
memcpy
(
work_val
,
val
,
map
->
format
.
val_bytes
);
val
=
work_val
;
}
if
(
map
->
async
&&
map
->
bus
->
async_write
)
{
struct
regmap_async
*
async
;
trace_regmap_async_write_start
(
map
->
dev
,
reg
,
val_len
);
trace_regmap_async_write_start
(
map
->
dev
,
reg
,
val_len
);
async
->
work_buf
=
kzalloc
(
map
->
format
.
buf_size
,
spin_lock_irqsave
(
&
map
->
async_lock
,
flags
);
GFP_KERNEL
|
GFP_DMA
);
async
=
list_first_entry_or_null
(
&
map
->
async_free
,
if
(
!
async
->
work_buf
)
{
struct
regmap_async
,
kfree
(
async
);
list
);
return
-
ENOMEM
;
if
(
async
)
list_del
(
&
async
->
list
);
spin_unlock_irqrestore
(
&
map
->
async_lock
,
flags
);
if
(
!
async
)
{
async
=
map
->
bus
->
async_alloc
();
if
(
!
async
)
return
-
ENOMEM
;
async
->
work_buf
=
kzalloc
(
map
->
format
.
buf_size
,
GFP_KERNEL
|
GFP_DMA
);
if
(
!
async
->
work_buf
)
{
kfree
(
async
);
return
-
ENOMEM
;
}
}
}
INIT_WORK
(
&
async
->
cleanup
,
async_cleanup
);
async
->
map
=
map
;
async
->
map
=
map
;
/* If the caller supplied the value we can use it safely. */
/* If the caller supplied the value we can use it safely. */
memcpy
(
async
->
work_buf
,
map
->
work_buf
,
map
->
format
.
pad_bytes
+
memcpy
(
async
->
work_buf
,
map
->
work_buf
,
map
->
format
.
pad_bytes
+
map
->
format
.
reg_bytes
+
map
->
format
.
val_bytes
);
map
->
format
.
reg_bytes
+
map
->
format
.
val_bytes
);
if
(
val
==
work_val
)
val
=
async
->
work_buf
+
map
->
format
.
pad_bytes
+
map
->
format
.
reg_bytes
;
spin_lock_irqsave
(
&
map
->
async_lock
,
flags
);
spin_lock_irqsave
(
&
map
->
async_lock
,
flags
);
list_add_tail
(
&
async
->
list
,
&
map
->
async_list
);
list_add_tail
(
&
async
->
list
,
&
map
->
async_list
);
spin_unlock_irqrestore
(
&
map
->
async_lock
,
flags
);
spin_unlock_irqrestore
(
&
map
->
async_lock
,
flags
);
ret
=
map
->
bus
->
async_write
(
map
->
bus_context
,
async
->
work_buf
,
if
(
val
!=
work_val
)
map
->
format
.
reg_bytes
+
ret
=
map
->
bus
->
async_write
(
map
->
bus_context
,
map
->
format
.
pad_bytes
,
async
->
work_buf
,
val
,
val_len
,
async
);
map
->
format
.
reg_bytes
+
map
->
format
.
pad_bytes
,
val
,
val_len
,
async
);
else
ret
=
map
->
bus
->
async_write
(
map
->
bus_context
,
async
->
work_buf
,
map
->
format
.
reg_bytes
+
map
->
format
.
pad_bytes
+
val_len
,
NULL
,
0
,
async
);
if
(
ret
!=
0
)
{
if
(
ret
!=
0
)
{
dev_err
(
map
->
dev
,
"Failed to schedule write: %d
\n
"
,
dev_err
(
map
->
dev
,
"Failed to schedule write: %d
\n
"
,
ret
);
ret
);
spin_lock_irqsave
(
&
map
->
async_lock
,
flags
);
spin_lock_irqsave
(
&
map
->
async_lock
,
flags
);
list_
del
(
&
async
->
list
);
list_
move
(
&
async
->
list
,
&
map
->
async_free
);
spin_unlock_irqrestore
(
&
map
->
async_lock
,
flags
);
spin_unlock_irqrestore
(
&
map
->
async_lock
,
flags
);
kfree
(
async
->
work_buf
);
kfree
(
async
);
}
}
return
ret
;
return
ret
;
...
@@ -1253,7 +1278,7 @@ static int _regmap_bus_raw_write(void *context, unsigned int reg,
...
@@ -1253,7 +1278,7 @@ static int _regmap_bus_raw_write(void *context, unsigned int reg,
map
->
work_buf
+
map
->
work_buf
+
map
->
format
.
reg_bytes
+
map
->
format
.
reg_bytes
+
map
->
format
.
pad_bytes
,
map
->
format
.
pad_bytes
,
map
->
format
.
val_bytes
,
false
);
map
->
format
.
val_bytes
);
}
}
static
inline
void
*
_regmap_map_get_context
(
struct
regmap
*
map
)
static
inline
void
*
_regmap_map_get_context
(
struct
regmap
*
map
)
...
@@ -1317,6 +1342,37 @@ int regmap_write(struct regmap *map, unsigned int reg, unsigned int val)
...
@@ -1317,6 +1342,37 @@ int regmap_write(struct regmap *map, unsigned int reg, unsigned int val)
}
}
EXPORT_SYMBOL_GPL
(
regmap_write
);
EXPORT_SYMBOL_GPL
(
regmap_write
);
/**
* regmap_write_async(): Write a value to a single register asynchronously
*
* @map: Register map to write to
* @reg: Register to write to
* @val: Value to be written
*
* A value of zero will be returned on success, a negative errno will
* be returned in error cases.
*/
int
regmap_write_async
(
struct
regmap
*
map
,
unsigned
int
reg
,
unsigned
int
val
)
{
int
ret
;
if
(
reg
%
map
->
reg_stride
)
return
-
EINVAL
;
map
->
lock
(
map
->
lock_arg
);
map
->
async
=
true
;
ret
=
_regmap_write
(
map
,
reg
,
val
);
map
->
async
=
false
;
map
->
unlock
(
map
->
lock_arg
);
return
ret
;
}
EXPORT_SYMBOL_GPL
(
regmap_write_async
);
/**
/**
* regmap_raw_write(): Write raw values to one or more registers
* regmap_raw_write(): Write raw values to one or more registers
*
*
...
@@ -1345,7 +1401,7 @@ int regmap_raw_write(struct regmap *map, unsigned int reg,
...
@@ -1345,7 +1401,7 @@ int regmap_raw_write(struct regmap *map, unsigned int reg,
map
->
lock
(
map
->
lock_arg
);
map
->
lock
(
map
->
lock_arg
);
ret
=
_regmap_raw_write
(
map
,
reg
,
val
,
val_len
,
false
);
ret
=
_regmap_raw_write
(
map
,
reg
,
val
,
val_len
);
map
->
unlock
(
map
->
lock_arg
);
map
->
unlock
(
map
->
lock_arg
);
...
@@ -1421,14 +1477,12 @@ int regmap_bulk_write(struct regmap *map, unsigned int reg, const void *val,
...
@@ -1421,14 +1477,12 @@ int regmap_bulk_write(struct regmap *map, unsigned int reg, const void *val,
ret
=
_regmap_raw_write
(
map
,
ret
=
_regmap_raw_write
(
map
,
reg
+
(
i
*
map
->
reg_stride
),
reg
+
(
i
*
map
->
reg_stride
),
val
+
(
i
*
val_bytes
),
val
+
(
i
*
val_bytes
),
val_bytes
,
val_bytes
);
false
);
if
(
ret
!=
0
)
if
(
ret
!=
0
)
return
ret
;
return
ret
;
}
}
}
else
{
}
else
{
ret
=
_regmap_raw_write
(
map
,
reg
,
wval
,
val_bytes
*
val_count
,
ret
=
_regmap_raw_write
(
map
,
reg
,
wval
,
val_bytes
*
val_count
);
false
);
}
}
if
(
val_bytes
!=
1
)
if
(
val_bytes
!=
1
)
...
@@ -1474,7 +1528,11 @@ int regmap_raw_write_async(struct regmap *map, unsigned int reg,
...
@@ -1474,7 +1528,11 @@ int regmap_raw_write_async(struct regmap *map, unsigned int reg,
map
->
lock
(
map
->
lock_arg
);
map
->
lock
(
map
->
lock_arg
);
ret
=
_regmap_raw_write
(
map
,
reg
,
val
,
val_len
,
true
);
map
->
async
=
true
;
ret
=
_regmap_raw_write
(
map
,
reg
,
val
,
val_len
);
map
->
async
=
false
;
map
->
unlock
(
map
->
lock_arg
);
map
->
unlock
(
map
->
lock_arg
);
...
@@ -1788,6 +1846,41 @@ int regmap_update_bits(struct regmap *map, unsigned int reg,
...
@@ -1788,6 +1846,41 @@ int regmap_update_bits(struct regmap *map, unsigned int reg,
}
}
EXPORT_SYMBOL_GPL
(
regmap_update_bits
);
EXPORT_SYMBOL_GPL
(
regmap_update_bits
);
/**
* regmap_update_bits_async: Perform a read/modify/write cycle on the register
* map asynchronously
*
* @map: Register map to update
* @reg: Register to update
* @mask: Bitmask to change
* @val: New value for bitmask
*
* With most buses the read must be done synchronously so this is most
* useful for devices with a cache which do not need to interact with
* the hardware to determine the current register value.
*
* Returns zero for success, a negative number on error.
*/
int
regmap_update_bits_async
(
struct
regmap
*
map
,
unsigned
int
reg
,
unsigned
int
mask
,
unsigned
int
val
)
{
bool
change
;
int
ret
;
map
->
lock
(
map
->
lock_arg
);
map
->
async
=
true
;
ret
=
_regmap_update_bits
(
map
,
reg
,
mask
,
val
,
&
change
);
map
->
async
=
false
;
map
->
unlock
(
map
->
lock_arg
);
return
ret
;
}
EXPORT_SYMBOL_GPL
(
regmap_update_bits_async
);
/**
/**
* regmap_update_bits_check: Perform a read/modify/write cycle on the
* regmap_update_bits_check: Perform a read/modify/write cycle on the
* register map and report if updated
* register map and report if updated
...
@@ -1813,6 +1906,43 @@ int regmap_update_bits_check(struct regmap *map, unsigned int reg,
...
@@ -1813,6 +1906,43 @@ int regmap_update_bits_check(struct regmap *map, unsigned int reg,
}
}
EXPORT_SYMBOL_GPL
(
regmap_update_bits_check
);
EXPORT_SYMBOL_GPL
(
regmap_update_bits_check
);
/**
* regmap_update_bits_check_async: Perform a read/modify/write cycle on the
* register map asynchronously and report if
* updated
*
* @map: Register map to update
* @reg: Register to update
* @mask: Bitmask to change
* @val: New value for bitmask
* @change: Boolean indicating if a write was done
*
* With most buses the read must be done synchronously so this is most
* useful for devices with a cache which do not need to interact with
* the hardware to determine the current register value.
*
* Returns zero for success, a negative number on error.
*/
int
regmap_update_bits_check_async
(
struct
regmap
*
map
,
unsigned
int
reg
,
unsigned
int
mask
,
unsigned
int
val
,
bool
*
change
)
{
int
ret
;
map
->
lock
(
map
->
lock_arg
);
map
->
async
=
true
;
ret
=
_regmap_update_bits
(
map
,
reg
,
mask
,
val
,
change
);
map
->
async
=
false
;
map
->
unlock
(
map
->
lock_arg
);
return
ret
;
}
EXPORT_SYMBOL_GPL
(
regmap_update_bits_check_async
);
void
regmap_async_complete_cb
(
struct
regmap_async
*
async
,
int
ret
)
void
regmap_async_complete_cb
(
struct
regmap_async
*
async
,
int
ret
)
{
{
struct
regmap
*
map
=
async
->
map
;
struct
regmap
*
map
=
async
->
map
;
...
@@ -1821,8 +1951,7 @@ void regmap_async_complete_cb(struct regmap_async *async, int ret)
...
@@ -1821,8 +1951,7 @@ void regmap_async_complete_cb(struct regmap_async *async, int ret)
trace_regmap_async_io_complete
(
map
->
dev
);
trace_regmap_async_io_complete
(
map
->
dev
);
spin_lock
(
&
map
->
async_lock
);
spin_lock
(
&
map
->
async_lock
);
list_move
(
&
async
->
list
,
&
map
->
async_free
);
list_del
(
&
async
->
list
);
wake
=
list_empty
(
&
map
->
async_list
);
wake
=
list_empty
(
&
map
->
async_list
);
if
(
ret
!=
0
)
if
(
ret
!=
0
)
...
@@ -1830,8 +1959,6 @@ void regmap_async_complete_cb(struct regmap_async *async, int ret)
...
@@ -1830,8 +1959,6 @@ void regmap_async_complete_cb(struct regmap_async *async, int ret)
spin_unlock
(
&
map
->
async_lock
);
spin_unlock
(
&
map
->
async_lock
);
schedule_work
(
&
async
->
cleanup
);
if
(
wake
)
if
(
wake
)
wake_up
(
&
map
->
async_waitq
);
wake_up
(
&
map
->
async_waitq
);
}
}
...
@@ -1907,6 +2034,7 @@ int regmap_register_patch(struct regmap *map, const struct reg_default *regs,
...
@@ -1907,6 +2034,7 @@ int regmap_register_patch(struct regmap *map, const struct reg_default *regs,
bypass
=
map
->
cache_bypass
;
bypass
=
map
->
cache_bypass
;
map
->
cache_bypass
=
true
;
map
->
cache_bypass
=
true
;
map
->
async
=
true
;
/* Write out first; it's useful to apply even if we fail later. */
/* Write out first; it's useful to apply even if we fail later. */
for
(
i
=
0
;
i
<
num_regs
;
i
++
)
{
for
(
i
=
0
;
i
<
num_regs
;
i
++
)
{
...
@@ -1930,10 +2058,13 @@ int regmap_register_patch(struct regmap *map, const struct reg_default *regs,
...
@@ -1930,10 +2058,13 @@ int regmap_register_patch(struct regmap *map, const struct reg_default *regs,
}
}
out:
out:
map
->
async
=
false
;
map
->
cache_bypass
=
bypass
;
map
->
cache_bypass
=
bypass
;
map
->
unlock
(
map
->
lock_arg
);
map
->
unlock
(
map
->
lock_arg
);
regmap_async_complete
(
map
);
return
ret
;
return
ret
;
}
}
EXPORT_SYMBOL_GPL
(
regmap_register_patch
);
EXPORT_SYMBOL_GPL
(
regmap_register_patch
);
...
...
include/linux/regmap.h
View file @
5eff79fe
...
@@ -374,6 +374,7 @@ int regmap_reinit_cache(struct regmap *map,
...
@@ -374,6 +374,7 @@ int regmap_reinit_cache(struct regmap *map,
const
struct
regmap_config
*
config
);
const
struct
regmap_config
*
config
);
struct
regmap
*
dev_get_regmap
(
struct
device
*
dev
,
const
char
*
name
);
struct
regmap
*
dev_get_regmap
(
struct
device
*
dev
,
const
char
*
name
);
int
regmap_write
(
struct
regmap
*
map
,
unsigned
int
reg
,
unsigned
int
val
);
int
regmap_write
(
struct
regmap
*
map
,
unsigned
int
reg
,
unsigned
int
val
);
int
regmap_write_async
(
struct
regmap
*
map
,
unsigned
int
reg
,
unsigned
int
val
);
int
regmap_raw_write
(
struct
regmap
*
map
,
unsigned
int
reg
,
int
regmap_raw_write
(
struct
regmap
*
map
,
unsigned
int
reg
,
const
void
*
val
,
size_t
val_len
);
const
void
*
val
,
size_t
val_len
);
int
regmap_bulk_write
(
struct
regmap
*
map
,
unsigned
int
reg
,
const
void
*
val
,
int
regmap_bulk_write
(
struct
regmap
*
map
,
unsigned
int
reg
,
const
void
*
val
,
...
@@ -387,9 +388,14 @@ int regmap_bulk_read(struct regmap *map, unsigned int reg, void *val,
...
@@ -387,9 +388,14 @@ int regmap_bulk_read(struct regmap *map, unsigned int reg, void *val,
size_t
val_count
);
size_t
val_count
);
int
regmap_update_bits
(
struct
regmap
*
map
,
unsigned
int
reg
,
int
regmap_update_bits
(
struct
regmap
*
map
,
unsigned
int
reg
,
unsigned
int
mask
,
unsigned
int
val
);
unsigned
int
mask
,
unsigned
int
val
);
int
regmap_update_bits_async
(
struct
regmap
*
map
,
unsigned
int
reg
,
unsigned
int
mask
,
unsigned
int
val
);
int
regmap_update_bits_check
(
struct
regmap
*
map
,
unsigned
int
reg
,
int
regmap_update_bits_check
(
struct
regmap
*
map
,
unsigned
int
reg
,
unsigned
int
mask
,
unsigned
int
val
,
unsigned
int
mask
,
unsigned
int
val
,
bool
*
change
);
bool
*
change
);
int
regmap_update_bits_check_async
(
struct
regmap
*
map
,
unsigned
int
reg
,
unsigned
int
mask
,
unsigned
int
val
,
bool
*
change
);
int
regmap_get_val_bytes
(
struct
regmap
*
map
);
int
regmap_get_val_bytes
(
struct
regmap
*
map
);
int
regmap_async_complete
(
struct
regmap
*
map
);
int
regmap_async_complete
(
struct
regmap
*
map
);
bool
regmap_can_raw_write
(
struct
regmap
*
map
);
bool
regmap_can_raw_write
(
struct
regmap
*
map
);
...
@@ -527,6 +533,13 @@ static inline int regmap_write(struct regmap *map, unsigned int reg,
...
@@ -527,6 +533,13 @@ static inline int regmap_write(struct regmap *map, unsigned int reg,
return
-
EINVAL
;
return
-
EINVAL
;
}
}
static
inline
int
regmap_write_async
(
struct
regmap
*
map
,
unsigned
int
reg
,
unsigned
int
val
)
{
WARN_ONCE
(
1
,
"regmap API is disabled"
);
return
-
EINVAL
;
}
static
inline
int
regmap_raw_write
(
struct
regmap
*
map
,
unsigned
int
reg
,
static
inline
int
regmap_raw_write
(
struct
regmap
*
map
,
unsigned
int
reg
,
const
void
*
val
,
size_t
val_len
)
const
void
*
val
,
size_t
val_len
)
{
{
...
@@ -576,6 +589,14 @@ static inline int regmap_update_bits(struct regmap *map, unsigned int reg,
...
@@ -576,6 +589,14 @@ static inline int regmap_update_bits(struct regmap *map, unsigned int reg,
return
-
EINVAL
;
return
-
EINVAL
;
}
}
static
inline
int
regmap_update_bits_async
(
struct
regmap
*
map
,
unsigned
int
reg
,
unsigned
int
mask
,
unsigned
int
val
)
{
WARN_ONCE
(
1
,
"regmap API is disabled"
);
return
-
EINVAL
;
}
static
inline
int
regmap_update_bits_check
(
struct
regmap
*
map
,
static
inline
int
regmap_update_bits_check
(
struct
regmap
*
map
,
unsigned
int
reg
,
unsigned
int
reg
,
unsigned
int
mask
,
unsigned
int
val
,
unsigned
int
mask
,
unsigned
int
val
,
...
@@ -585,6 +606,16 @@ static inline int regmap_update_bits_check(struct regmap *map,
...
@@ -585,6 +606,16 @@ static inline int regmap_update_bits_check(struct regmap *map,
return
-
EINVAL
;
return
-
EINVAL
;
}
}
static
inline
int
regmap_update_bits_check_async
(
struct
regmap
*
map
,
unsigned
int
reg
,
unsigned
int
mask
,
unsigned
int
val
,
bool
*
change
)
{
WARN_ONCE
(
1
,
"regmap API is disabled"
);
return
-
EINVAL
;
}
static
inline
int
regmap_get_val_bytes
(
struct
regmap
*
map
)
static
inline
int
regmap_get_val_bytes
(
struct
regmap
*
map
)
{
{
WARN_ONCE
(
1
,
"regmap API is disabled"
);
WARN_ONCE
(
1
,
"regmap API is disabled"
);
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment