Permalink
Find file Copy path
Fetching contributors…
Cannot retrieve contributors at this time
650 lines (569 sloc) 23.8 KB
/*
* Copyright (c) 2007, Michael Feathers, James Grenning and Bas Vodde
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are met:
* * Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* * Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
* * Neither the name of the <organization> nor the
* names of its contributors may be used to endorse or promote products
* derived from this software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE EARLIER MENTIONED AUTHORS ``AS IS'' AND ANY
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
* DISCLAIMED. IN NO EVENT SHALL <copyright holder> BE LIABLE FOR ANY
* DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
#include "CppUTest/TestHarness.h"
#include "CppUTest/MemoryLeakDetector.h"
#include "CppUTest/TestMemoryAllocator.h"
#include "CppUTest/PlatformSpecificFunctions.h"
class MemoryLeakFailureForTest: public MemoryLeakFailure
{
public:
virtual ~MemoryLeakFailureForTest()
{
}
virtual void fail(char* fail_string)
{
*message = fail_string;
}
SimpleString *message;
};
class NewAllocatorForMemoryLeakDetectionTest: public TestMemoryAllocator
{
public:
NewAllocatorForMemoryLeakDetectionTest() :
TestMemoryAllocator("Standard New Allocator", "new", "delete"),
alloc_called(0), free_called(0)
{
}
int alloc_called;
int free_called;
char* alloc_memory(size_t size, const char*, int)
{
alloc_called++;
return TestMemoryAllocator::alloc_memory(size, "file", 1);
}
void free_memory(char* memory, const char* file, int line)
{
free_called++;
TestMemoryAllocator::free_memory(memory, file, line);
}
};
class AllocatorForMemoryLeakDetectionTest: public TestMemoryAllocator
{
public:
AllocatorForMemoryLeakDetectionTest() :
alloc_called(0), free_called(0), allocMemoryLeakNodeCalled(0), freeMemoryLeakNodeCalled(0)
{
}
int alloc_called;
int free_called;
int allocMemoryLeakNodeCalled;
int freeMemoryLeakNodeCalled;
char* alloc_memory(size_t size, const char* file, int line)
{
alloc_called++;
return TestMemoryAllocator::alloc_memory(size, file, line);
}
void free_memory(char* memory, const char* file, int line)
{
free_called++;
TestMemoryAllocator::free_memory(memory, file, line);
}
char* allocMemoryLeakNode(size_t size)
{
allocMemoryLeakNodeCalled++;
return TestMemoryAllocator::alloc_memory(size, __FILE__, __LINE__);
}
void freeMemoryLeakNode(char* memory)
{
freeMemoryLeakNodeCalled++;
TestMemoryAllocator::free_memory(memory, __FILE__, __LINE__);
}
};
TEST_GROUP(MemoryLeakDetectorTest)
{
MemoryLeakDetector* detector;
MemoryLeakFailureForTest *reporter;
AllocatorForMemoryLeakDetectionTest* testAllocator;
void setup()
{
reporter = new MemoryLeakFailureForTest;
detector = new MemoryLeakDetector(reporter);
testAllocator = new AllocatorForMemoryLeakDetectionTest;
detector->enable();
detector->startChecking();
reporter->message = new SimpleString();
}
void teardown()
{
delete reporter->message;
delete detector;
delete reporter;
delete testAllocator;
}
};
TEST(MemoryLeakDetectorTest, OneLeak)
{
char* mem = detector->allocMemory(testAllocator, 3);
detector->stopChecking();
SimpleString output = detector->report(mem_leak_period_checking);
STRCMP_CONTAINS("Memory leak(s) found", output.asCharString());
STRCMP_CONTAINS("size: 3", output.asCharString());
STRCMP_CONTAINS("alloc", output.asCharString());
STRCMP_CONTAINS(StringFromFormat("%p", (void*) mem).asCharString(), output.asCharString());
STRCMP_CONTAINS("Total number of leaks", output.asCharString());
PlatformSpecificFree(mem);
LONGS_EQUAL(1, testAllocator->alloc_called);
LONGS_EQUAL(0, testAllocator->free_called);
}
TEST(MemoryLeakDetectorTest, sequenceNumbersOfMemoryLeaks)
{
char* mem = detector->allocMemory(defaultNewAllocator(), 1);
char* mem2 = detector->allocMemory(defaultNewAllocator(), 2);
char* mem3 = detector->allocMemory(defaultNewAllocator(), 3);
SimpleString output = detector->report(mem_leak_period_checking);
STRCMP_CONTAINS("Alloc num (1)", output.asCharString());
STRCMP_CONTAINS("Alloc num (2)", output.asCharString());
STRCMP_CONTAINS("Alloc num (3)", output.asCharString());
PlatformSpecificFree(mem);
PlatformSpecificFree(mem2);
PlatformSpecificFree(mem3);
}
TEST(MemoryLeakDetectorTest, memoryDumpOutput)
{
char* mem = detector->allocMemory(defaultNewAllocator(), 6);
SimpleString::StrNCpy(mem, "test1", 6);
SimpleString output = detector->report(mem_leak_period_checking);
STRCMP_CONTAINS("Alloc num (1)", output.asCharString());
STRCMP_CONTAINS("Leak size: 6 Allocated at", output.asCharString());
STRCMP_CONTAINS("Content:", output.asCharString());
STRCMP_CONTAINS("0000: 74 65 73 74 31 00 |test1.|", output.asCharString());
PlatformSpecificFree(mem);
}
TEST(MemoryLeakDetectorTest, OneHundredLeaks)
{
const int amount_alloc = 100;
char *mem[amount_alloc];
for (int i = 0; i < amount_alloc; i++)
mem[i] = detector->allocMemory(defaultMallocAllocator(), 3);
detector->stopChecking();
SimpleString output = detector->report(mem_leak_period_checking);
STRCMP_CONTAINS("Memory leak(s) found", output.asCharString());
STRCMP_CONTAINS("Total number of leaks", output.asCharString());
STRCMP_CONTAINS("Memory leak reports about malloc and free", output.asCharString());
//don't reuse i for vc6 compatibility
for (int j = 0; j < amount_alloc; j++)
PlatformSpecificFree(mem[j]);
}
TEST(MemoryLeakDetectorTest, OneLeakOutsideCheckingPeriod)
{
detector->stopChecking();
char* mem = detector->allocMemory(defaultNewAllocator(), 4);
SimpleString output = detector->report(mem_leak_period_all);
CHECK(output.contains("Memory leak(s) found"));
CHECK(output.contains("size: 4"));
CHECK(output.contains("new"));
CHECK(output.contains("Total number of leaks"));
PlatformSpecificFree(mem);
}
TEST(MemoryLeakDetectorTest, NoLeaksWhatsoever)
{
detector->stopChecking();
STRCMP_CONTAINS("No memory leaks", detector->report(mem_leak_period_checking));
STRCMP_CONTAINS("No memory leaks", detector->report(mem_leak_period_all));
}
TEST(MemoryLeakDetectorTest, TwoLeaksUsingOperatorNew)
{
char* mem = detector->allocMemory(defaultNewAllocator(), 4);
char* mem2 = detector->allocMemory(defaultNewAllocator(), 8);
detector->stopChecking();
SimpleString output = detector->report(mem_leak_period_checking);
LONGS_EQUAL(2, detector->totalMemoryLeaks(mem_leak_period_checking));
CHECK(output.contains("size: 8"));
CHECK(output.contains("size: 4"));
PlatformSpecificFree(mem);
PlatformSpecificFree(mem2);
}
TEST(MemoryLeakDetectorTest, OneAllocButNoLeak)
{
char* mem = detector->allocMemory(testAllocator, 4);
detector->deallocMemory(testAllocator, mem);
detector->stopChecking();
STRCMP_CONTAINS("No memory leaks", detector->report(mem_leak_period_checking));
LONGS_EQUAL(1, testAllocator->alloc_called);
LONGS_EQUAL(1, testAllocator->free_called);
}
TEST(MemoryLeakDetectorTest, TwoAllocOneFreeOneLeak)
{
char* mem = detector->allocMemory(testAllocator, 4);
char* mem2 = detector->allocMemory(testAllocator, 12);
detector->deallocMemory(testAllocator, mem);
detector->stopChecking();
SimpleString output = detector->report(mem_leak_period_checking);
LONGS_EQUAL(1, detector->totalMemoryLeaks(mem_leak_period_checking));
CHECK(output.contains("Leak size: 12"));
CHECK(!output.contains("Leak size: 4"));
PlatformSpecificFree(mem2);
LONGS_EQUAL(2, testAllocator->alloc_called);
LONGS_EQUAL(1, testAllocator->free_called);
}
TEST(MemoryLeakDetectorTest, TwoAllocOneFreeOneLeakReverseOrder)
{
char* mem = detector->allocMemory(defaultNewAllocator(), 4);
char* mem2 = detector->allocMemory(defaultNewAllocator(), 12);
detector->deallocMemory(defaultNewAllocator(), mem2);
detector->stopChecking();
SimpleString output = detector->report(mem_leak_period_checking);
LONGS_EQUAL(1, detector->totalMemoryLeaks(mem_leak_period_checking));
CHECK(!output.contains("size: 12"));
CHECK(output.contains("size: 4"));
PlatformSpecificFree(mem);
}
TEST(MemoryLeakDetectorTest, DeleteNonAlocatedMemory)
{
char a;
char* pa = &a;
detector->deallocMemory(defaultMallocAllocator(), pa, "FREE.c", 100);
detector->stopChecking();
CHECK(reporter->message->contains("Deallocating non-allocated memory"));
CHECK(reporter->message->contains(" allocated at file: <unknown> line: 0 size: 0 type: unknown"));
CHECK(reporter->message->contains(" deallocated at file: FREE.c line: 100 type: free"));
}
TEST(MemoryLeakDetectorTest, IgnoreMemoryAllocatedOutsideCheckingPeriod)
{
detector->stopChecking();
char* mem = detector->allocMemory(defaultNewAllocator(), 4);
LONGS_EQUAL(0, detector->totalMemoryLeaks(mem_leak_period_checking));
LONGS_EQUAL(1, detector->totalMemoryLeaks(mem_leak_period_all));
detector->deallocMemory(defaultNewAllocator(), mem);
}
TEST(MemoryLeakDetectorTest, IgnoreMemoryAllocatedOutsideCheckingPeriodComplicatedCase)
{
char* mem = detector->allocMemory(defaultNewAllocator(), 4);
detector->stopChecking();
char* mem2 = detector->allocMemory(defaultNewAllocator(), 8);
LONGS_EQUAL(1, detector->totalMemoryLeaks(mem_leak_period_checking));
detector->clearAllAccounting(mem_leak_period_checking);
PlatformSpecificFree(mem);
LONGS_EQUAL(0, detector->totalMemoryLeaks(mem_leak_period_checking));
LONGS_EQUAL(1, detector->totalMemoryLeaks(mem_leak_period_all));
detector->startChecking();
char* mem3 = detector->allocMemory(defaultNewAllocator(), 4);
detector->stopChecking();
LONGS_EQUAL(1, detector->totalMemoryLeaks(mem_leak_period_checking));
LONGS_EQUAL(2, detector->totalMemoryLeaks(mem_leak_period_all));
detector->clearAllAccounting(mem_leak_period_checking);
LONGS_EQUAL(0, detector->totalMemoryLeaks(mem_leak_period_checking));
LONGS_EQUAL(1, detector->totalMemoryLeaks(mem_leak_period_all));
detector->clearAllAccounting(mem_leak_period_all);
LONGS_EQUAL(0, detector->totalMemoryLeaks(mem_leak_period_all));
PlatformSpecificFree(mem2);
PlatformSpecificFree(mem3);
}
TEST(MemoryLeakDetectorTest, OneLeakUsingOperatorNewWithFileLine)
{
char* mem = detector->allocMemory(defaultNewAllocator(), 4, "file.cpp", 1234);
detector->stopChecking();
SimpleString output = detector->report(mem_leak_period_checking);
CHECK(output.contains("file.cpp"));
CHECK(output.contains("1234"));
PlatformSpecificFree(mem);
}
TEST(MemoryLeakDetectorTest, OneAllocAndFreeUsingArrayNew)
{
char* mem = detector->allocMemory(defaultNewArrayAllocator(), 10, "file.cpp", 1234);
char* mem2 = detector->allocMemory(defaultNewArrayAllocator(), 12);
LONGS_EQUAL(2, detector->totalMemoryLeaks(mem_leak_period_all));
SimpleString output = detector->report(mem_leak_period_checking);
CHECK(output.contains("new []"));
detector->deallocMemory(defaultNewArrayAllocator(), mem);
detector->deallocMemory(defaultNewArrayAllocator(), mem2);
LONGS_EQUAL(0, detector->totalMemoryLeaks(mem_leak_period_all));
detector->stopChecking();
}
TEST(MemoryLeakDetectorTest, OneAllocAndFree)
{
char* mem = detector->allocMemory(defaultMallocAllocator(), 10, "file.cpp", 1234);
char* mem2 = detector->allocMemory(defaultMallocAllocator(), 12);
LONGS_EQUAL(2, detector->totalMemoryLeaks(mem_leak_period_checking));
SimpleString output = detector->report(mem_leak_period_checking);
CHECK(output.contains("malloc"));
detector->deallocMemory(defaultMallocAllocator(), mem);
detector->deallocMemory(defaultMallocAllocator(), mem2, "file.c", 5678);
LONGS_EQUAL(0, detector->totalMemoryLeaks(mem_leak_period_all));
detector->stopChecking();
}
TEST(MemoryLeakDetectorTest, OneRealloc)
{
char* mem1 = detector->allocMemory(testAllocator, 10, "file.cpp", 1234, true);
char* mem2 = detector->reallocMemory(testAllocator, mem1, 1000, "other.cpp", 5678, true);
LONGS_EQUAL(1, detector->totalMemoryLeaks(mem_leak_period_checking));
SimpleString output = detector->report(mem_leak_period_checking);
CHECK(output.contains("other.cpp"));
detector->deallocMemory(testAllocator, mem2, true);
LONGS_EQUAL(0, detector->totalMemoryLeaks(mem_leak_period_all));
detector->stopChecking();
LONGS_EQUAL(1, testAllocator->alloc_called);
LONGS_EQUAL(1, testAllocator->free_called);
LONGS_EQUAL(2, testAllocator->allocMemoryLeakNodeCalled);
LONGS_EQUAL(2, testAllocator->freeMemoryLeakNodeCalled);
}
TEST(MemoryLeakDetectorTest, ReallocNonAllocatedMemory)
{
char mem1;
char* mem2 = detector->reallocMemory(testAllocator, &mem1, 5, "other.cpp", 13, true);
detector->deallocMemory(testAllocator, mem2, true);
detector->stopChecking();
CHECK(reporter->message->contains("Deallocating non-allocated memory\n"));
CHECK(reporter->message->contains(" deallocated at file: other.cpp line: 13"));
}
TEST(MemoryLeakDetectorTest, AllocOneTypeFreeAnotherType)
{
char* mem = detector->allocMemory(defaultNewArrayAllocator(), 100, "ALLOC.c", 10);
detector->deallocMemory(defaultMallocAllocator(), mem, "FREE.c", 100);
detector->stopChecking();
CHECK(reporter->message->contains("Allocation/deallocation type mismatch"));
CHECK(reporter->message->contains(" allocated at file: ALLOC.c line: 10 size: 100 type: new []"));
CHECK(reporter->message->contains(" deallocated at file: FREE.c line: 100 type: free"));
}
TEST(MemoryLeakDetectorTest, AllocOneTypeFreeAnotherTypeWithCheckingDisabled)
{
detector->disableAllocationTypeChecking();
char* mem = detector->allocMemory(defaultNewArrayAllocator(), 100, "ALLOC.c", 10);
detector->deallocMemory(defaultNewAllocator(), mem, "FREE.c", 100);
detector->stopChecking();
STRCMP_EQUAL("", reporter->message->asCharString());
detector->enableAllocationTypeChecking();
}
TEST(MemoryLeakDetectorTest, mallocLeakGivesAdditionalWarning)
{
char* mem = detector->allocMemory(defaultMallocAllocator(), 100, "ALLOC.c", 10);
detector->stopChecking();
SimpleString output = detector->report(mem_leak_period_checking);
STRCMP_CONTAINS("Memory leak reports about malloc and free can be caused by allocating using the cpputest version of malloc", output.asCharString());
PlatformSpecificFree(mem);
}
TEST(MemoryLeakDetectorTest, newLeakDoesNotGiveAdditionalWarning)
{
char* mem = detector->allocMemory(defaultNewAllocator(), 100, "ALLOC.c", 10);
detector->stopChecking();
SimpleString output = detector->report(mem_leak_period_checking);
CHECK(! output.contains("Memory leak reports about malloc and free"));
PlatformSpecificFree(mem);
}
TEST(MemoryLeakDetectorTest, MarkCheckingPeriodLeaksAsNonCheckingPeriod)
{
char* mem = detector->allocMemory(defaultNewArrayAllocator(), 100);
char* mem2 = detector->allocMemory(defaultNewArrayAllocator(), 100);
detector->stopChecking();
LONGS_EQUAL(2, detector->totalMemoryLeaks(mem_leak_period_checking));
LONGS_EQUAL(2, detector->totalMemoryLeaks(mem_leak_period_all));
detector->markCheckingPeriodLeaksAsNonCheckingPeriod();
LONGS_EQUAL(0, detector->totalMemoryLeaks(mem_leak_period_checking));
LONGS_EQUAL(2, detector->totalMemoryLeaks(mem_leak_period_all));
PlatformSpecificFree(mem);
PlatformSpecificFree(mem2);
}
TEST(MemoryLeakDetectorTest, memoryCorruption)
{
char* mem = detector->allocMemory(defaultMallocAllocator(), 10, "ALLOC.c", 10);
mem[10] = 'O';
mem[11] = 'H';
detector->deallocMemory(defaultMallocAllocator(), mem, "FREE.c", 100);
detector->stopChecking();
CHECK(reporter->message->contains("Memory corruption"));
CHECK(reporter->message->contains(" allocated at file: ALLOC.c line: 10 size: 10 type: malloc"));
CHECK(reporter->message->contains(" deallocated at file: FREE.c line: 100 type: free"));
}
TEST(MemoryLeakDetectorTest, safelyDeleteNULL)
{
detector->deallocMemory(defaultNewAllocator(), NULLPTR);
STRCMP_EQUAL("", reporter->message->asCharString());
}
TEST(MemoryLeakDetectorTest, periodDisabled)
{
detector->disable();
char* mem = detector->allocMemory(defaultMallocAllocator(), 2);
LONGS_EQUAL(1, detector->totalMemoryLeaks(mem_leak_period_all));
LONGS_EQUAL(1, detector->totalMemoryLeaks(mem_leak_period_disabled));
LONGS_EQUAL(0, detector->totalMemoryLeaks(mem_leak_period_enabled));
LONGS_EQUAL(0, detector->totalMemoryLeaks(mem_leak_period_checking));
detector->deallocMemory(defaultMallocAllocator(), mem);
}
TEST(MemoryLeakDetectorTest, periodEnabled)
{
detector->enable();
char* mem = detector->allocMemory(defaultMallocAllocator(), 2);
LONGS_EQUAL(1, detector->totalMemoryLeaks(mem_leak_period_all));
LONGS_EQUAL(0, detector->totalMemoryLeaks(mem_leak_period_disabled));
LONGS_EQUAL(1, detector->totalMemoryLeaks(mem_leak_period_enabled));
LONGS_EQUAL(0, detector->totalMemoryLeaks(mem_leak_period_checking));
detector->deallocMemory(defaultMallocAllocator(), mem);
}
TEST(MemoryLeakDetectorTest, periodChecking)
{
char* mem = detector->allocMemory(defaultMallocAllocator(), 2);
LONGS_EQUAL(1, detector->totalMemoryLeaks(mem_leak_period_all));
LONGS_EQUAL(0, detector->totalMemoryLeaks(mem_leak_period_disabled));
LONGS_EQUAL(1, detector->totalMemoryLeaks(mem_leak_period_enabled));
LONGS_EQUAL(1, detector->totalMemoryLeaks(mem_leak_period_checking));
detector->deallocMemory(defaultMallocAllocator(), mem);
}
TEST(MemoryLeakDetectorTest, allocateWithANullAllocatorCausesNoProblems)
{
char* mem = detector->allocMemory(NullUnknownAllocator::defaultAllocator(), 2);
detector->deallocMemory(NullUnknownAllocator::defaultAllocator(), mem);
}
TEST(MemoryLeakDetectorTest, invalidateMemory)
{
unsigned char* mem = (unsigned char*)detector->allocMemory(defaultMallocAllocator(), 2);
detector->invalidateMemory((char*)mem);
CHECK(mem[0] == 0xCD);
CHECK(mem[1] == 0xCD);
detector->deallocMemory(defaultMallocAllocator(), mem);
}
TEST(MemoryLeakDetectorTest, invalidateMemoryNULLShouldWork)
{
detector->invalidateMemory(NULLPTR);
}
TEST_GROUP(MemoryLeakDetectorListTest)
{
};
TEST(MemoryLeakDetectorListTest, clearAllAccountingIsWorkingProperly)
{
MemoryLeakDetectorList listForTesting;
MemoryLeakDetectorNode node1, node2, node3;
node3.period_ = mem_leak_period_disabled;
listForTesting.addNewNode(&node1);
listForTesting.addNewNode(&node2);
listForTesting.addNewNode(&node3);
listForTesting.clearAllAccounting(mem_leak_period_enabled);
POINTERS_EQUAL(NULLPTR, listForTesting.getFirstLeak(mem_leak_period_enabled));
CHECK(&node3 == listForTesting.getFirstLeak(mem_leak_period_disabled));
}
TEST_GROUP(SimpleStringBuffer)
{
};
TEST(SimpleStringBuffer, initialStringIsEmpty)
{
SimpleStringBuffer buffer;
STRCMP_EQUAL("", buffer.toString());
}
TEST(SimpleStringBuffer, simpleTest)
{
SimpleStringBuffer buffer;
buffer.add("Hello");
buffer.add(" World");
STRCMP_EQUAL("Hello World", buffer.toString());
}
TEST(SimpleStringBuffer, writePastLimit)
{
SimpleStringBuffer buffer;
for (int i = 0; i < SimpleStringBuffer::SIMPLE_STRING_BUFFER_LEN * 2; i++)
buffer.add("h");
SimpleString str("h", SimpleStringBuffer::SIMPLE_STRING_BUFFER_LEN-1);
STRCMP_EQUAL(str.asCharString(), buffer.toString());
}
TEST(SimpleStringBuffer, setWriteLimit)
{
SimpleStringBuffer buffer;
buffer.setWriteLimit(10);
for (int i = 0; i < SimpleStringBuffer::SIMPLE_STRING_BUFFER_LEN ; i++)
buffer.add("h");
SimpleString str("h", 10);
STRCMP_EQUAL(str.asCharString(), buffer.toString());
}
TEST(SimpleStringBuffer, setWriteLimitTooHighIsIgnored)
{
SimpleStringBuffer buffer;
buffer.setWriteLimit(SimpleStringBuffer::SIMPLE_STRING_BUFFER_LEN+10);
for (int i = 0; i < SimpleStringBuffer::SIMPLE_STRING_BUFFER_LEN+10; i++)
buffer.add("h");
SimpleString str("h", SimpleStringBuffer::SIMPLE_STRING_BUFFER_LEN-1);
STRCMP_EQUAL(str.asCharString(), buffer.toString());
}
TEST(SimpleStringBuffer, resetWriteLimit)
{
SimpleStringBuffer buffer;
buffer.setWriteLimit(10);
for (int i = 0; i < SimpleStringBuffer::SIMPLE_STRING_BUFFER_LEN ; i++)
buffer.add("h");
buffer.resetWriteLimit();
buffer.add("%s", SimpleString("h", 10).asCharString());
SimpleString str("h", 20);
STRCMP_EQUAL(str.asCharString(), buffer.toString());
}
TEST(SimpleStringBuffer, addMemoryDumpOneLinePlusOnePartial)
{
SimpleStringBuffer buffer;
buffer.addMemoryDump("deadbeefdeadbeefhopsxx", 22);
STRCMP_EQUAL(" 0000: 64 65 61 64 62 65 65 66 64 65 61 64 62 65 65 66 |deadbeefdeadbeef|\n"
" 0010: 68 6f 70 73 78 78 |hopsxx|\n",
buffer.toString());
}
TEST(SimpleStringBuffer, addMemoryDumpNonPrintable)
{
SimpleStringBuffer buffer;
// Ensure we test edge cases - NUL, 0x1F, 0x7F, 0xFF
buffer.addMemoryDump("\x15\x7f\xff\x00\x1ftdd", 8);
STRCMP_EQUAL(" 0000: 15 7f ff 00 1f 74 64 64 |.....tdd|\n",
buffer.toString());
}
TEST(SimpleStringBuffer, addMemoryDumpOneLine)
{
SimpleStringBuffer buffer;
buffer.addMemoryDump("deadbeefdeadbeef", 16);
STRCMP_EQUAL(" 0000: 64 65 61 64 62 65 65 66 64 65 61 64 62 65 65 66 |deadbeefdeadbeef|\n",
buffer.toString());
}
TEST(SimpleStringBuffer, addMemoryDumpOneHalfLine)
{
SimpleStringBuffer buffer;
buffer.addMemoryDump("deadbeef", 8);
STRCMP_EQUAL(" 0000: 64 65 61 64 62 65 65 66 |deadbeef|\n",
buffer.toString());
}
TEST(SimpleStringBuffer, addMemoryDumpOneByte)
{
SimpleStringBuffer buffer;
buffer.addMemoryDump("Z", 1);
STRCMP_EQUAL(" 0000: 5a |Z|\n",
buffer.toString());
}
TEST(SimpleStringBuffer, addMemoryDumpZeroBytes)
{
SimpleStringBuffer buffer;
buffer.addMemoryDump("", 0);
STRCMP_EQUAL("", buffer.toString());
}
TEST_GROUP(ReallocBugReported)
{
MemoryLeakFailureForTest reporter;
};
TEST(ReallocBugReported, CanSafelyDoAReallocWithANewAllocator)
{
MemoryLeakDetector detector(&reporter);
char* mem = detector.allocMemory(getCurrentNewAllocator(), 5, "file", 1);
mem = detector.reallocMemory(getCurrentNewAllocator(), mem, 19, "file", 1);
detector.deallocMemory(getCurrentNewAllocator(), mem);
}
TEST(ReallocBugReported, CanSafelyDoAReallocWithAMallocAllocator)
{
MemoryLeakDetector detector(&reporter);
char* mem = detector.allocMemory(getCurrentMallocAllocator(), 5, "file", 1, true);
mem = detector.reallocMemory(getCurrentMallocAllocator(), mem, 19, "file", 1, true);
detector.deallocMemory(getCurrentMallocAllocator(), mem, true);
}