Skip to content
GitLab
Explore
Sign in
Primary navigation
Search or go to…
Project
K
KED Linux Kernel Fork
Manage
Activity
Members
Labels
Plan
Issues
Issue boards
Milestones
Code
Merge requests
Repository
Branches
Commits
Tags
Repository graph
Compare revisions
Deploy
Model registry
Analyze
Model experiments
Help
Help
Support
GitLab documentation
Compare GitLab plans
Community forum
Contribute to GitLab
Provide feedback
Keyboard shortcuts
?
Snippets
Groups
Projects
Show more breadcrumbs
KED Software Projects
Miscellaneous
KED Linux Kernel Fork
Commits
1db7c89c
Commit
1db7c89c
authored
13 years ago
by
Mark Brown
Browse files
Options
Downloads
Plain Diff
Merge branch 'for-3.2' into for-3.3
parents
7b282cbb
bda63586
No related branches found
Branches containing commit
No related tags found
Tags containing commit
No related merge requests found
Changes
2
Hide whitespace changes
Inline
Side-by-side
Showing
2 changed files
drivers/firmware/sigma.c
+58
-23
58 additions, 23 deletions
drivers/firmware/sigma.c
include/linux/sigma.h
+4
-9
4 additions, 9 deletions
include/linux/sigma.h
with
62 additions
and
32 deletions
drivers/firmware/sigma.c
+
58
−
23
View file @
1db7c89c
...
@@ -14,13 +14,34 @@
...
@@ -14,13 +14,34 @@
#include
<linux/module.h>
#include
<linux/module.h>
#include
<linux/sigma.h>
#include
<linux/sigma.h>
/* Return: 0==OK, <0==error, =1 ==no more actions */
static
size_t
sigma_action_size
(
struct
sigma_action
*
sa
)
{
size_t
payload
=
0
;
switch
(
sa
->
instr
)
{
case
SIGMA_ACTION_WRITEXBYTES
:
case
SIGMA_ACTION_WRITESINGLE
:
case
SIGMA_ACTION_WRITESAFELOAD
:
payload
=
sigma_action_len
(
sa
);
break
;
default:
break
;
}
payload
=
ALIGN
(
payload
,
2
);
return
payload
+
sizeof
(
struct
sigma_action
);
}
/*
* Returns a negative error value in case of an error, 0 if processing of
* the firmware should be stopped after this action, 1 otherwise.
*/
static
int
static
int
process_sigma_action
(
struct
i2c_client
*
client
,
struct
sigma_
firmware
*
ssfw
)
process_sigma_action
(
struct
i2c_client
*
client
,
struct
sigma_
action
*
sa
)
{
{
struct
sigma_action
*
sa
=
(
void
*
)(
ssfw
->
fw
->
data
+
ssfw
->
pos
);
size_t
len
=
sigma_action_len
(
sa
);
size_t
len
=
sigma_action_len
(
sa
);
int
ret
=
0
;
int
ret
;
pr_debug
(
"%s: instr:%i addr:%#x len:%zu
\n
"
,
__func__
,
pr_debug
(
"%s: instr:%i addr:%#x len:%zu
\n
"
,
__func__
,
sa
->
instr
,
sa
->
addr
,
len
);
sa
->
instr
,
sa
->
addr
,
len
);
...
@@ -29,44 +50,50 @@ process_sigma_action(struct i2c_client *client, struct sigma_firmware *ssfw)
...
@@ -29,44 +50,50 @@ process_sigma_action(struct i2c_client *client, struct sigma_firmware *ssfw)
case
SIGMA_ACTION_WRITEXBYTES
:
case
SIGMA_ACTION_WRITEXBYTES
:
case
SIGMA_ACTION_WRITESINGLE
:
case
SIGMA_ACTION_WRITESINGLE
:
case
SIGMA_ACTION_WRITESAFELOAD
:
case
SIGMA_ACTION_WRITESAFELOAD
:
if
(
ssfw
->
fw
->
size
<
ssfw
->
pos
+
len
)
return
-
EINVAL
;
ret
=
i2c_master_send
(
client
,
(
void
*
)
&
sa
->
addr
,
len
);
ret
=
i2c_master_send
(
client
,
(
void
*
)
&
sa
->
addr
,
len
);
if
(
ret
<
0
)
if
(
ret
<
0
)
return
-
EINVAL
;
return
-
EINVAL
;
break
;
break
;
case
SIGMA_ACTION_DELAY
:
case
SIGMA_ACTION_DELAY
:
ret
=
0
;
udelay
(
len
);
udelay
(
len
);
len
=
0
;
len
=
0
;
break
;
break
;
case
SIGMA_ACTION_END
:
case
SIGMA_ACTION_END
:
return
1
;
return
0
;
default:
default:
return
-
EINVAL
;
return
-
EINVAL
;
}
}
/* when arrive here ret=0 or sent data */
return
1
;
ssfw
->
pos
+=
sigma_action_size
(
sa
,
len
);
return
ssfw
->
pos
==
ssfw
->
fw
->
size
;
}
}
static
int
static
int
process_sigma_actions
(
struct
i2c_client
*
client
,
struct
sigma_firmware
*
ssfw
)
process_sigma_actions
(
struct
i2c_client
*
client
,
struct
sigma_firmware
*
ssfw
)
{
{
pr_debug
(
"%s: processing %p
\n
"
,
__func__
,
ssfw
);
struct
sigma_action
*
sa
;
size_t
size
;
int
ret
;
while
(
ssfw
->
pos
+
sizeof
(
*
sa
)
<=
ssfw
->
fw
->
size
)
{
sa
=
(
struct
sigma_action
*
)(
ssfw
->
fw
->
data
+
ssfw
->
pos
);
size
=
sigma_action_size
(
sa
);
ssfw
->
pos
+=
size
;
if
(
ssfw
->
pos
>
ssfw
->
fw
->
size
||
size
==
0
)
break
;
ret
=
process_sigma_action
(
client
,
sa
);
while
(
1
)
{
int
ret
=
process_sigma_action
(
client
,
ssfw
);
pr_debug
(
"%s: action returned %i
\n
"
,
__func__
,
ret
);
pr_debug
(
"%s: action returned %i
\n
"
,
__func__
,
ret
);
if
(
ret
==
1
)
return
0
;
if
(
ret
<=
0
)
else
if
(
ret
)
return
ret
;
return
ret
;
}
}
if
(
ssfw
->
pos
!=
ssfw
->
fw
->
size
)
return
-
EINVAL
;
return
0
;
}
}
int
process_sigma_firmware
(
struct
i2c_client
*
client
,
const
char
*
name
)
int
process_sigma_firmware
(
struct
i2c_client
*
client
,
const
char
*
name
)
...
@@ -89,16 +116,24 @@ int process_sigma_firmware(struct i2c_client *client, const char *name)
...
@@ -89,16 +116,24 @@ int process_sigma_firmware(struct i2c_client *client, const char *name)
/* then verify the header */
/* then verify the header */
ret
=
-
EINVAL
;
ret
=
-
EINVAL
;
if
(
fw
->
size
<
sizeof
(
*
ssfw_head
))
/*
* Reject too small or unreasonable large files. The upper limit has been
* chosen a bit arbitrarily, but it should be enough for all practical
* purposes and having the limit makes it easier to avoid integer
* overflows later in the loading process.
*/
if
(
fw
->
size
<
sizeof
(
*
ssfw_head
)
||
fw
->
size
>=
0x4000000
)
goto
done
;
goto
done
;
ssfw_head
=
(
void
*
)
fw
->
data
;
ssfw_head
=
(
void
*
)
fw
->
data
;
if
(
memcmp
(
ssfw_head
->
magic
,
SIGMA_MAGIC
,
ARRAY_SIZE
(
ssfw_head
->
magic
)))
if
(
memcmp
(
ssfw_head
->
magic
,
SIGMA_MAGIC
,
ARRAY_SIZE
(
ssfw_head
->
magic
)))
goto
done
;
goto
done
;
crc
=
crc32
(
0
,
fw
->
data
,
fw
->
size
);
crc
=
crc32
(
0
,
fw
->
data
+
sizeof
(
*
ssfw_head
),
fw
->
size
-
sizeof
(
*
ssfw_head
));
pr_debug
(
"%s: crc=%x
\n
"
,
__func__
,
crc
);
pr_debug
(
"%s: crc=%x
\n
"
,
__func__
,
crc
);
if
(
crc
!=
ssfw_head
->
crc
)
if
(
crc
!=
le32_to_cpu
(
ssfw_head
->
crc
)
)
goto
done
;
goto
done
;
ssfw
.
pos
=
sizeof
(
*
ssfw_head
);
ssfw
.
pos
=
sizeof
(
*
ssfw_head
);
...
...
This diff is collapsed.
Click to expand it.
include/linux/sigma.h
+
4
−
9
View file @
1db7c89c
...
@@ -24,7 +24,7 @@ struct sigma_firmware {
...
@@ -24,7 +24,7 @@ struct sigma_firmware {
struct
sigma_firmware_header
{
struct
sigma_firmware_header
{
unsigned
char
magic
[
7
];
unsigned
char
magic
[
7
];
u8
version
;
u8
version
;
u
32
crc
;
__le
32
crc
;
};
};
enum
{
enum
{
...
@@ -40,19 +40,14 @@ enum {
...
@@ -40,19 +40,14 @@ enum {
struct
sigma_action
{
struct
sigma_action
{
u8
instr
;
u8
instr
;
u8
len_hi
;
u8
len_hi
;
u
16
len
;
__le
16
len
;
u
16
addr
;
__be
16
addr
;
unsigned
char
payload
[];
unsigned
char
payload
[];
};
};
static
inline
u32
sigma_action_len
(
struct
sigma_action
*
sa
)
static
inline
u32
sigma_action_len
(
struct
sigma_action
*
sa
)
{
{
return
(
sa
->
len_hi
<<
16
)
|
sa
->
len
;
return
(
sa
->
len_hi
<<
16
)
|
le16_to_cpu
(
sa
->
len
);
}
static
inline
size_t
sigma_action_size
(
struct
sigma_action
*
sa
,
u32
payload_len
)
{
return
sizeof
(
*
sa
)
+
payload_len
+
(
payload_len
%
2
);
}
}
extern
int
process_sigma_firmware
(
struct
i2c_client
*
client
,
const
char
*
name
);
extern
int
process_sigma_firmware
(
struct
i2c_client
*
client
,
const
char
*
name
);
...
...
This diff is collapsed.
Click to expand it.
Preview
0%
Loading
Try again
or
attach a new file
.
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Save comment
Cancel
Please
register
or
sign in
to comment