[rtems commit] bsp/arm: Correct cache misalignment handling

Sebastian Huber sebh at rtems.org
Thu Apr 17 11:21:10 UTC 2014


Module:    rtems
Branch:    master
Commit:    db5a84d0ad4186e8875cdad9db15c7516364921f
Changeset: http://git.rtems.org/rtems/commit/?id=db5a84d0ad4186e8875cdad9db15c7516364921f

Author:    Ralf Kirchner <ralf.kirchner at embedded-brains.de>
Date:      Thu Apr 17 10:59:47 2014 +0200

bsp/arm: Correct cache misalignment handling

Correct misalignment handling and prepare for locking.

---

 c/src/lib/libbsp/arm/shared/arm-l2c-310/cache_.h   |   80 +++++++++++++-------
 c/src/lib/libbsp/arm/shared/include/arm-cache-l1.h |   12 ++--
 2 files changed, 60 insertions(+), 32 deletions(-)

diff --git a/c/src/lib/libbsp/arm/shared/arm-l2c-310/cache_.h b/c/src/lib/libbsp/arm/shared/arm-l2c-310/cache_.h
index 322c7a4..0c3133b 100644
--- a/c/src/lib/libbsp/arm/shared/arm-l2c-310/cache_.h
+++ b/c/src/lib/libbsp/arm/shared/arm-l2c-310/cache_.h
@@ -1120,22 +1120,17 @@ cache_l2c_310_invalidate_1_line( const void *d_addr )
 }
 
 static inline void
-cache_l2c_310_invalidate_range( const void *addr, size_t n_bytes )
+cache_l2c_310_invalidate_range( uint32_t adx, const uint32_t ADDR_LAST )
 {
-  if ( n_bytes != 0 ) {
-    uint32_t       adx  = (uint32_t) addr
-                         & ~CACHE_L2C_310_INSTRUCTION_LINE_MASK;
-    const uint32_t end  =
-      ( adx + n_bytes ) & ~CACHE_L2C_310_INSTRUCTION_LINE_MASK;
     volatile L2CC *l2cc = (volatile L2CC *) BSP_ARM_L2CC_BASE;
 
     /* Back starting address up to start of a line and invalidate until end */
-    for (;
-         adx < end;
-         adx += CPU_INSTRUCTION_CACHE_ALIGNMENT ) {
-      /* Invalidate L2 cache line */
-      l2cc->inv_pa = adx;
-    }
+  for (;
+       adx <= ADDR_LAST;
+       adx += CPU_INSTRUCTION_CACHE_ALIGNMENT ) {
+    /* Invalidate L2 cache line */
+    l2cc->inv_pa = adx;
+  }
     cache_l2c_310_sync();
   }
 }
@@ -1407,19 +1402,41 @@ _CPU_cache_invalidate_data_range(
 )
 {
   if ( n_bytes > 0 ) {
+    /* Back starting address up to start of a line and invalidate until ADDR_LAST */
+    uint32_t       adx       = (uint32_t) addr_first
+      & ~CACHE_L2C_310_DATA_LINE_MASK;
+    const uint32_t ADDR_LAST =
+      (uint32_t)( (size_t)addr_first + n_bytes - 1 );
+    uint32_t       block_end =
+      CACHE_MIN( ADDR_LAST, adx + CACHE_MAX_LOCKING_BYTES );
     
-    cache_l2c_310_invalidate_range(
-      addr_first,
-      n_bytes
-    );
+    /* We have to apply a lock. Thus we will operate only CACHE_MAX_LOCKING_BYTES
+     * at a time */
+    for (;
+         adx      <= ADDR_LAST;
+         adx       = block_end + 1,
+         block_end = CACHE_MIN( ADDR_LAST, adx + CACHE_MAX_LOCKING_BYTES )) {
+      cache_l2c_310_invalidate_range(
+        adx,
+        block_end
+      );
+    }
     arm_cache_l1_invalidate_data_range(
       addr_first,
       n_bytes
     );
-    cache_l2c_310_invalidate_range(
-      addr_first,
-      n_bytes
-    );
+
+    adx       = (uint32_t)addr_first & ~CACHE_L2C_310_DATA_LINE_MASK;
+    block_end = CACHE_MIN( ADDR_LAST, adx + CACHE_MAX_LOCKING_BYTES );
+    for (;
+         adx      <= ADDR_LAST;
+         adx       = block_end + 1,
+         block_end = CACHE_MIN( ADDR_LAST, adx + CACHE_MAX_LOCKING_BYTES )) {
+      cache_l2c_310_invalidate_range(
+        adx,
+        block_end
+      );
+    }
     arm_cache_l1_invalidate_data_range(
       addr_first,
       n_bytes
@@ -1471,13 +1488,24 @@ _CPU_cache_invalidate_instruction_range(
 )
 {
   if ( n_bytes != 0 ) {
-    
+    uint32_t       adx       = (uint32_t) i_addr
+      & ~CACHE_L2C_310_DATA_LINE_MASK;
+    const uint32_t ADDR_LAST =
+    (uint32_t)( (size_t)i_addr + n_bytes - 1 );
+    uint32_t       block_end =
+      CACHE_MIN( ADDR_LAST, adx + CACHE_MAX_LOCKING_BYTES );
+
     /* Invalidate L2 cache lines */
-    cache_l2c_310_invalidate_range(
-      i_addr,
-      n_bytes
-    );
-    
+    for (;
+         adx      <= ADDR_LAST;
+         adx       = block_end + 1,
+         block_end = CACHE_MIN( ADDR_LAST, adx + CACHE_MAX_LOCKING_BYTES )) {
+      cache_l2c_310_invalidate_range(
+        adx,
+        block_end
+      );
+    }
+
     arm_cache_l1_invalidate_instruction_range(
       i_addr,
       n_bytes
diff --git a/c/src/lib/libbsp/arm/shared/include/arm-cache-l1.h b/c/src/lib/libbsp/arm/shared/include/arm-cache-l1.h
index 77069bd..e4a19ad 100644
--- a/c/src/lib/libbsp/arm/shared/include/arm-cache-l1.h
+++ b/c/src/lib/libbsp/arm/shared/include/arm-cache-l1.h
@@ -247,8 +247,8 @@ static inline void arm_cache_l1_flush_data_range(
     uint32_t       adx       = (uint32_t) d_addr
                                & ~ARM_CACHE_L1_DATA_LINE_MASK;
     const uint32_t ADDR_LAST =
-      ( (uint32_t) d_addr + n_bytes - 1 ) & ~ARM_CACHE_L1_DATA_LINE_MASK;
-      
+      (uint32_t)( (size_t) d_addr + n_bytes - 1 );
+
     ARM_CACHE_L1_ERRATA_764369_HANDLER();
 
     for (; adx <= ADDR_LAST; adx += ARM_CACHE_L1_CPU_DATA_ALIGNMENT ) {
@@ -300,13 +300,13 @@ static inline void arm_cache_l1_invalidate_data_range(
     uint32_t       adx = (uint32_t) d_addr
                          & ~ARM_CACHE_L1_DATA_LINE_MASK;
     const uint32_t end =
-      ( adx + n_bytes ) & ~ARM_CACHE_L1_DATA_LINE_MASK;
+      (uint32_t)( (size_t)d_addr + n_bytes -1);
 
     ARM_CACHE_L1_ERRATA_764369_HANDLER();
     
     /* Back starting address up to start of a line and invalidate until end */
     for (;
-         adx < end;
+         adx <= end;
          adx += ARM_CACHE_L1_CPU_DATA_ALIGNMENT ) {
         /* Invalidate the Instruction cache line */
         arm_cp15_data_cache_invalidate_line( (void*)adx );
@@ -325,7 +325,7 @@ static inline void arm_cache_l1_invalidate_instruction_range(
     uint32_t       adx = (uint32_t) i_addr
                          & ~ARM_CACHE_L1_INSTRUCTION_LINE_MASK;
     const uint32_t end =
-      ( adx + n_bytes ) & ~ARM_CACHE_L1_INSTRUCTION_LINE_MASK;
+      (uint32_t)( (size_t)i_addr + n_bytes -1);
 
     arm_cache_l1_select( ARM_CACHE_L1_CSS_ID_INSTRUCTION );
 
@@ -333,7 +333,7 @@ static inline void arm_cache_l1_invalidate_instruction_range(
 
     /* Back starting address up to start of a line and invalidate until end */
     for (;
-         adx < end;
+         adx <= end;
          adx += ARM_CACHE_L1_CPU_INSTRUCTION_ALIGNMENT ) {
         /* Invalidate the Instruction cache line */
         arm_cp15_instruction_cache_invalidate_line( (void*)adx );




More information about the vc mailing list