Skip to content
GitLab
Explore
Sign in
Primary navigation
Search or go to…
Project
R
RIOT
Manage
Activity
Members
Labels
Plan
Issues
Issue boards
Milestones
Wiki
Code
Merge requests
Repository
Branches
Commits
Tags
Repository graph
Compare revisions
Snippets
Build
Pipelines
Jobs
Pipeline schedules
Artifacts
Deploy
Releases
Container Registry
Model registry
Operate
Environments
Monitor
Incidents
Service Desk
Analyze
Value stream analytics
Contributor analytics
CI/CD analytics
Repository analytics
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
cm-projects
RIOT
Commits
6be334b4
Commit
6be334b4
authored
8 years ago
by
Joakim Nohlgård
Committed by
GitHub
8 years ago
Browse files
Options
Downloads
Plain Diff
Merge pull request #5833 from immesys/fix-i2c-checks
samd21: i2c: check all busstate bits
parents
0a15d1b2
e367ab93
No related branches found
Branches containing commit
No related tags found
No related merge requests found
Changes
1
Hide whitespace changes
Inline
Side-by-side
Showing
1 changed file
cpu/samd21/periph/i2c.c
+10
-5
10 additions, 5 deletions
cpu/samd21/periph/i2c.c
with
10 additions
and
5 deletions
cpu/samd21/periph/i2c.c
+
10
−
5
View file @
6be334b4
...
@@ -35,6 +35,11 @@
...
@@ -35,6 +35,11 @@
#define SAMD21_I2C_TIMEOUT (65535)
#define SAMD21_I2C_TIMEOUT (65535)
#define BUSSTATE_UNKNOWN SERCOM_I2CM_STATUS_BUSSTATE(0)
#define BUSSTATE_IDLE SERCOM_I2CM_STATUS_BUSSTATE(1)
#define BUSSTATE_OWNER SERCOM_I2CM_STATUS_BUSSTATE(2)
#define BUSSTATE_BUSY SERCOM_I2CM_STATUS_BUSSTATE(3)
/* static function definitions */
/* static function definitions */
static
void
_i2c_poweron
(
SercomI2cm
*
sercom
);
static
void
_i2c_poweron
(
SercomI2cm
*
sercom
);
static
void
_i2c_poweroff
(
SercomI2cm
*
sercom
);
static
void
_i2c_poweroff
(
SercomI2cm
*
sercom
);
...
@@ -197,10 +202,10 @@ int i2c_init_master(i2c_t dev, i2c_speed_t speed)
...
@@ -197,10 +202,10 @@ int i2c_init_master(i2c_t dev, i2c_speed_t speed)
i2c_poweron
(
dev
);
i2c_poweron
(
dev
);
/* Start timeout if bus state is unknown. */
/* Start timeout if bus state is unknown. */
while
(
!
(
I2CSercom
->
STATUS
.
reg
&
SERCOM_I2CM_STATUS_BUSSTATE
(
1
))
)
{
while
((
I2CSercom
->
STATUS
.
reg
&
SERCOM_I2CM_STATUS_BUSSTATE
_Msk
)
==
BUSSTATE_UNKNOWN
)
{
if
(
timeout_counter
++
>=
SAMD21_I2C_TIMEOUT
)
{
if
(
timeout_counter
++
>=
SAMD21_I2C_TIMEOUT
)
{
/* Timeout, force bus state to idle. */
/* Timeout, force bus state to idle. */
I2CSercom
->
STATUS
.
reg
=
SERCOM_I2CM_STATUS_
BUSSTATE
(
1
)
;
I2CSercom
->
STATUS
.
reg
=
BUSSTATE
_IDLE
;
}
}
}
}
return
0
;
return
0
;
...
@@ -435,7 +440,7 @@ static inline int _write(SercomI2cm *dev, char *data, int length)
...
@@ -435,7 +440,7 @@ static inline int _write(SercomI2cm *dev, char *data, int length)
DEBUG
(
"Looping through bytes
\n
"
);
DEBUG
(
"Looping through bytes
\n
"
);
while
(
tmp_data_length
--
)
{
while
(
tmp_data_length
--
)
{
/* Check that bus ownership is not lost. */
/* Check that bus ownership is not lost. */
if
(
!
(
dev
->
STATUS
.
reg
&
SERCOM_I2CM_STATUS_BUSSTATE
(
2
))
)
{
if
((
dev
->
STATUS
.
reg
&
SERCOM_I2CM_STATUS_BUSSTATE
_Msk
)
!=
BUSSTATE_OWNER
)
{
DEBUG
(
"STATUS_ERR_PACKET_COLLISION
\n
"
);
DEBUG
(
"STATUS_ERR_PACKET_COLLISION
\n
"
);
return
-
2
;
return
-
2
;
}
}
...
@@ -476,7 +481,7 @@ static inline int _read(SercomI2cm *dev, char *data, int length)
...
@@ -476,7 +481,7 @@ static inline int _read(SercomI2cm *dev, char *data, int length)
/* Read data buffer. */
/* Read data buffer. */
while
(
count
!=
length
)
{
while
(
count
!=
length
)
{
/* Check that bus ownership is not lost. */
/* Check that bus ownership is not lost. */
if
(
!
(
dev
->
STATUS
.
reg
&
SERCOM_I2CM_STATUS_BUSSTATE
(
2
))
)
{
if
((
dev
->
STATUS
.
reg
&
SERCOM_I2CM_STATUS_BUSSTATE
_Msk
)
!=
BUSSTATE_OWNER
)
{
DEBUG
(
"STATUS_ERR_PACKET_COLLISION
\n
"
);
DEBUG
(
"STATUS_ERR_PACKET_COLLISION
\n
"
);
return
-
2
;
return
-
2
;
}
}
...
@@ -509,7 +514,7 @@ static inline void _stop(SercomI2cm *dev)
...
@@ -509,7 +514,7 @@ static inline void _stop(SercomI2cm *dev)
/* Stop command */
/* Stop command */
dev
->
CTRLB
.
reg
|=
SERCOM_I2CM_CTRLB_CMD
(
3
);
dev
->
CTRLB
.
reg
|=
SERCOM_I2CM_CTRLB_CMD
(
3
);
/* Wait for bus to be idle again */
/* Wait for bus to be idle again */
while
(
dev
->
STATUS
.
reg
&
SERCOM_I2CM_STATUS_BUSSTATE
(
1
)
)
{}
while
(
(
dev
->
STATUS
.
reg
&
SERCOM_I2CM_STATUS_BUSSTATE
_Msk
)
!=
BUSSTATE_IDLE
)
{}
DEBUG
(
"Stop sent
\n
"
);
DEBUG
(
"Stop sent
\n
"
);
}
}
...
...
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