8089a74233
Also fixing the ScopedComPtrVector unit test. The problem with the test was that it expected AddRef to be called twice and Release() once when pushing a pointer to the vector. However that only happens when a push causes a reallocation. When there's no reallocation, there's only a single call to AddRef and no call to Release, causing the test to report failure. The fix is to simply use std::list instead and fix the expected values. TEST=Run *ScopedComPtr* unit tests. BUG=none Review URL: http://codereview.chromium.org/149345 git-svn-id: svn://svn.chromium.org/chrome/trunk/src@20183 0039d316-1c4b-4281-b951-d872f2087c98
108 linhas
2.9 KiB
C++
108 linhas
2.9 KiB
C++
// Copyright (c) 2006-2008 The Chromium Authors. All rights reserved.
|
|
// Use of this source code is governed by a BSD-style license that can be
|
|
// found in the LICENSE file.
|
|
|
|
#include "base/scoped_comptr_win.h"
|
|
|
|
#include <shlobj.h>
|
|
|
|
#include "base/scoped_ptr.h"
|
|
#include "testing/gtest/include/gtest/gtest.h"
|
|
|
|
namespace {
|
|
|
|
struct Dummy {
|
|
Dummy() : adds(0), releases(0) { }
|
|
void AddRef() { ++adds; }
|
|
void Release() { ++releases; }
|
|
|
|
int adds;
|
|
int releases;
|
|
};
|
|
|
|
extern const IID dummy_iid;
|
|
const IID dummy_iid = { 0x12345678u, 0x1234u, 0x5678u, 01, 23, 45, 67, 89,
|
|
01, 23, 45 };
|
|
|
|
} // namespace
|
|
|
|
TEST(ScopedComPtrTest, ScopedComPtr) {
|
|
EXPECT_TRUE(memcmp(&ScopedComPtr<IUnknown>::iid(), &IID_IUnknown,
|
|
sizeof(IID)) == 0);
|
|
|
|
EXPECT_TRUE(SUCCEEDED(::CoInitialize(NULL)));
|
|
|
|
{
|
|
ScopedComPtr<IUnknown> unk;
|
|
EXPECT_TRUE(SUCCEEDED(unk.CreateInstance(CLSID_ShellLink)));
|
|
ScopedComPtr<IUnknown> unk2;
|
|
unk2.Attach(unk.Detach());
|
|
EXPECT_TRUE(unk == NULL);
|
|
EXPECT_TRUE(unk2 != NULL);
|
|
|
|
ScopedComPtr<IMalloc> mem_alloc;
|
|
EXPECT_TRUE(SUCCEEDED(CoGetMalloc(1, mem_alloc.Receive())));
|
|
|
|
ScopedComPtr<IUnknown> qi_test;
|
|
EXPECT_HRESULT_SUCCEEDED(mem_alloc.QueryInterface(IID_IUnknown,
|
|
reinterpret_cast<void**>(qi_test.Receive())));
|
|
EXPECT_TRUE(qi_test.get() != NULL);
|
|
qi_test.Release();
|
|
|
|
// test ScopedComPtr& constructor
|
|
ScopedComPtr<IMalloc> copy1(mem_alloc);
|
|
EXPECT_TRUE(copy1.IsSameObject(mem_alloc));
|
|
EXPECT_FALSE(copy1.IsSameObject(unk2)); // unk2 is valid but different
|
|
EXPECT_FALSE(copy1.IsSameObject(unk)); // unk is NULL
|
|
|
|
IMalloc* naked_copy = copy1.Detach();
|
|
copy1 = naked_copy; // Test the =(T*) operator.
|
|
naked_copy->Release();
|
|
|
|
copy1.Release();
|
|
EXPECT_FALSE(copy1.IsSameObject(unk2)); // unk2 is valid, copy1 is not
|
|
|
|
// test Interface* constructor
|
|
ScopedComPtr<IMalloc> copy2(static_cast<IMalloc*>(mem_alloc));
|
|
EXPECT_TRUE(copy2.IsSameObject(mem_alloc));
|
|
|
|
EXPECT_TRUE(SUCCEEDED(unk.QueryFrom(mem_alloc)));
|
|
EXPECT_TRUE(unk != NULL);
|
|
unk.Release();
|
|
EXPECT_TRUE(unk == NULL);
|
|
EXPECT_TRUE(unk.IsSameObject(copy1)); // both are NULL
|
|
}
|
|
|
|
::CoUninitialize();
|
|
}
|
|
|
|
TEST(ScopedComPtrTest, ScopedComPtrVector) {
|
|
// Verify we don't get error C2558.
|
|
typedef ScopedComPtr<Dummy, &dummy_iid> Ptr;
|
|
std::vector<Ptr> bleh;
|
|
|
|
scoped_ptr<Dummy> p(new Dummy);
|
|
{
|
|
Ptr p2(p.get());
|
|
EXPECT_EQ(p->adds, 1);
|
|
EXPECT_EQ(p->releases, 0);
|
|
Ptr p3 = p2;
|
|
EXPECT_EQ(p->adds, 2);
|
|
EXPECT_EQ(p->releases, 0);
|
|
p3 = p2;
|
|
EXPECT_EQ(p->adds, 3);
|
|
EXPECT_EQ(p->releases, 1);
|
|
// To avoid hitting a reallocation.
|
|
bleh.reserve(1);
|
|
bleh.push_back(p2);
|
|
EXPECT_EQ(p->adds, 4);
|
|
EXPECT_EQ(p->releases, 1);
|
|
EXPECT_EQ(bleh[0], p.get());
|
|
bleh.pop_back();
|
|
EXPECT_EQ(p->adds, 4);
|
|
EXPECT_EQ(p->releases, 2);
|
|
}
|
|
EXPECT_EQ(p->adds, 4);
|
|
EXPECT_EQ(p->releases, 4);
|
|
}
|