Author: bperroud
Date: Thu Jan  5 21:46:00 2012
New Revision: 1227854

URL: http://svn.apache.org/viewvc?rev=1227854&view=rev
Log:
DIRECTMEMORY-55 : OffHeapMemoryBuffer leaks 1 byte at every allocation, 
firstmatch to check correct pointer size, return null when allocating on 
multiple buffers, report correct used size

Added:
    
incubator/directmemory/trunk/directmemory-cache/src/test/java/org/apache/directmemory/memory/test/MemoryManagerServiceImplTest.java
Modified:
    
incubator/directmemory/trunk/directmemory-cache/src/main/java/org/apache/directmemory/memory/OffHeapMemoryBuffer.java
    
incubator/directmemory/trunk/directmemory-cache/src/main/java/org/apache/directmemory/memory/Pointer.java

Modified: 
incubator/directmemory/trunk/directmemory-cache/src/main/java/org/apache/directmemory/memory/OffHeapMemoryBuffer.java
URL: 
http://svn.apache.org/viewvc/incubator/directmemory/trunk/directmemory-cache/src/main/java/org/apache/directmemory/memory/OffHeapMemoryBuffer.java?rev=1227854&r1=1227853&r2=1227854&view=diff
==============================================================================
--- 
incubator/directmemory/trunk/directmemory-cache/src/main/java/org/apache/directmemory/memory/OffHeapMemoryBuffer.java
 (original)
+++ 
incubator/directmemory/trunk/directmemory-cache/src/main/java/org/apache/directmemory/memory/OffHeapMemoryBuffer.java
 Thu Jan  5 21:46:00 2012
@@ -99,9 +99,9 @@ public class OffHeapMemoryBuffer
         Pointer fresh = new Pointer();
         fresh.bufferNumber = existing.bufferNumber;
         fresh.start = existing.start;
-        fresh.end = fresh.start + capacity;
+        fresh.end = fresh.start + capacity - 1; // 0 indexed
         fresh.free = true;
-        existing.start += capacity + 1;
+        existing.start = fresh.end + 1; // more readable
         return fresh;
     }
 
@@ -110,7 +110,7 @@ public class OffHeapMemoryBuffer
     {
         for ( Pointer ptr : pointers )
         {
-            if ( ptr.free && ptr.end >= capacity )
+            if (ptr.free && ptr.getCapacity() >= capacity)
             {
                 return ptr;
             }
@@ -143,7 +143,7 @@ public class OffHeapMemoryBuffer
             }
         }
 
-        final byte[] swp = new byte[pointer.end - pointer.start];
+        final byte[] swp = new byte[pointer.getCapacity()];
         buf.get( swp );
         return swp;
     }
@@ -158,8 +158,8 @@ public class OffHeapMemoryBuffer
         pointer2free.expiresIn = 0;
         pointer2free.clazz = null;
         pointer2free.directBuffer = null;
-        used.addAndGet( -( pointer2free.end - pointer2free.start ) );
-        return pointer2free.end - pointer2free.start;
+        used.addAndGet( - pointer2free.getCapacity() );
+        return pointer2free.getCapacity();
     }
 
     public void clear()
@@ -188,8 +188,7 @@ public class OffHeapMemoryBuffer
         if ( goodOne == null )
         {
             allocationErrors++;
-            throw new NullPointerException(
-                "did not find a suitable buffer " + allocationErrors + " times 
since last cleanup" );
+            return null;
         }
 
         Pointer fresh = slice( goodOne, payload.length );
@@ -353,8 +352,7 @@ public class OffHeapMemoryBuffer
         if ( goodOne == null )
         {
             allocationErrors++;
-            throw new NullPointerException(
-                "did not find a suitable buffer " + allocationErrors + " times 
since last cleanup" );
+            return null;
         }
 
         Pointer fresh = slice( goodOne, size );

Modified: 
incubator/directmemory/trunk/directmemory-cache/src/main/java/org/apache/directmemory/memory/Pointer.java
URL: 
http://svn.apache.org/viewvc/incubator/directmemory/trunk/directmemory-cache/src/main/java/org/apache/directmemory/memory/Pointer.java?rev=1227854&r1=1227853&r2=1227854&view=diff
==============================================================================
--- 
incubator/directmemory/trunk/directmemory-cache/src/main/java/org/apache/directmemory/memory/Pointer.java
 (original)
+++ 
incubator/directmemory/trunk/directmemory-cache/src/main/java/org/apache/directmemory/memory/Pointer.java
 Thu Jan  5 21:46:00 2012
@@ -45,6 +45,16 @@ public class Pointer
 
     public ByteBuffer directBuffer = null;
 
+    public Pointer()
+    {
+    }
+
+    public Pointer( int start, int end )
+    {
+        this.start = start;
+        this.end = end;
+    }
+
     public byte[] content()
     {
         return null;
@@ -66,4 +76,15 @@ public class Pointer
     {
         return (float) ( System.currentTimeMillis() - created ) / hits;
     }
+
+    public int getCapacity()
+    {
+        return end - start + 1;
+    }
+
+    @Override
+    public String toString()
+    {
+        return getClass().getSimpleName() + "[" + start + "," + end + "]" + ( 
free ? "" : "not" ) + "free";
+    }
 }

Added: 
incubator/directmemory/trunk/directmemory-cache/src/test/java/org/apache/directmemory/memory/test/MemoryManagerServiceImplTest.java
URL: 
http://svn.apache.org/viewvc/incubator/directmemory/trunk/directmemory-cache/src/test/java/org/apache/directmemory/memory/test/MemoryManagerServiceImplTest.java?rev=1227854&view=auto
==============================================================================
--- 
incubator/directmemory/trunk/directmemory-cache/src/test/java/org/apache/directmemory/memory/test/MemoryManagerServiceImplTest.java
 (added)
+++ 
incubator/directmemory/trunk/directmemory-cache/src/test/java/org/apache/directmemory/memory/test/MemoryManagerServiceImplTest.java
 Thu Jan  5 21:46:00 2012
@@ -0,0 +1,125 @@
+package org.apache.directmemory.memory.test;
+
+/*
+ * Licensed to the Apache Software Foundation (ASF) under one
+ * or more contributor license agreements.  See the NOTICE file
+ * distributed with this work for additional information
+ * regarding copyright ownership.  The ASF licenses this file
+ * to you under the Apache License, Version 2.0 (the
+ * "License"); you may not use this file except in compliance
+ * with the License.  You may obtain a copy of the License at
+ *
+ *  http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing,
+ * software distributed under the License is distributed on an
+ * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
+ * KIND, either express or implied.  See the License for the
+ * specific language governing permissions and limitations
+ * under the License.
+ */
+
+import junit.framework.Assert;
+
+import org.apache.directmemory.memory.MemoryManagerServiceImpl;
+import org.apache.directmemory.memory.Pointer;
+import org.junit.Test;
+
+public class MemoryManagerServiceImplTest {
+       
+       private static final byte[] SMALL_PAYLOAD = "ABCD".getBytes();
+
+       @Test
+       public void testFirstMatchBorderCase() {
+               
+               // Storing a first payload of 4 bytes, 1 byte remaining in the 
buffer. When storing a second 4 bytes payload, an BufferOverflowException is 
thrown instead of an easy check.
+               
+               final int BUFFER_SIZE = 5;
+               
+               final MemoryManagerServiceImpl memoryManagerService = new 
MemoryManagerServiceImpl();
+               
+               memoryManagerService.init(1, BUFFER_SIZE);
+               
+               Pointer pointer1 = memoryManagerService.store(SMALL_PAYLOAD);
+               Assert.assertNotNull(pointer1);
+               
+               Pointer pointer2 = memoryManagerService.store(SMALL_PAYLOAD);
+               Assert.assertNull(pointer2);
+               
+       }
+       
+       @Test
+       public void testAllocateMultipleBuffers() {
+
+               // Initializing 4 buffers of 4 bytes, MemoryManagerService 
should search for available space in another buffer.
+               
+               final int NUMBER_OF_OBJECTS = 4;
+               
+               final MemoryManagerServiceImpl memoryManagerService = new 
MemoryManagerServiceImpl();
+               
+               memoryManagerService.init(NUMBER_OF_OBJECTS, 
SMALL_PAYLOAD.length);
+                
+               for (int i = 0; i < NUMBER_OF_OBJECTS; i++) {
+                       Pointer pointer = 
memoryManagerService.store(SMALL_PAYLOAD);
+                       Assert.assertNotNull(pointer);
+               }
+               
+               Pointer pointerNull = memoryManagerService.store(SMALL_PAYLOAD);
+               Assert.assertNull(pointerNull);
+       }
+       
+       
+       @Test
+       public void testByteLeaking() {
+               
+               // Initializing 1 buffer of 10*4 bytes, should be able to 
allocate 10 objects of 4 bytes.
+               
+               final int NUMBER_OF_OBJECTS = 10;
+               
+               final MemoryManagerServiceImpl memoryManagerService = new 
MemoryManagerServiceImpl(); 
+               memoryManagerService.init(1, NUMBER_OF_OBJECTS * 
SMALL_PAYLOAD.length);
+               
+               for (int i = 0; i < NUMBER_OF_OBJECTS; i++) {
+                       Pointer pointer = 
memoryManagerService.store(SMALL_PAYLOAD);
+                       Assert.assertNotNull(pointer);
+               }
+               
+               Pointer pointerNull = memoryManagerService.store(SMALL_PAYLOAD);
+               Assert.assertNull(pointerNull);
+       }
+       
+       
+       @Test
+       public void testReportCorrectUsedMemory() {
+
+               // Initializing 1 buffer of 4*4 bytes, storing and freeing and 
storing again should report correct numbers.
+               
+               final int NUMBER_OF_OBJECTS = 4;
+               final int BUFFER_SIZE = NUMBER_OF_OBJECTS * 
SMALL_PAYLOAD.length;
+
+               final MemoryManagerServiceImpl memoryManagerService = new 
MemoryManagerServiceImpl();
+               
+               memoryManagerService.init(1, BUFFER_SIZE);
+               
+               Pointer lastPointer = null;
+               for (int i = 0; i < NUMBER_OF_OBJECTS; i++) {
+                       Pointer pointer = 
memoryManagerService.store(SMALL_PAYLOAD);
+                       Assert.assertNotNull(pointer);
+                       lastPointer = pointer;
+               }
+               
+               // Buffer is fully used.
+               Assert.assertEquals(BUFFER_SIZE, 
memoryManagerService.getBuffers().get(0).used());
+
+               Assert.assertNotNull(lastPointer);
+               memoryManagerService.free( lastPointer );
+               
+               Pointer pointerNotNull = 
memoryManagerService.store(SMALL_PAYLOAD);
+               Assert.assertNotNull(pointerNotNull);
+               
+               // Buffer again fully used.
+               Assert.assertEquals(BUFFER_SIZE, 
memoryManagerService.getBuffers().get(0).used());
+               
+       }
+       
+}


Reply via email to