/* | |
* Copyright (C) 2006 The Android Open Source Project | |
* | |
* Licensed 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. | |
*/ | |
/** \file | |
This file consists of implementation of a class AdbObjectHandle that | |
encapsulates an internal API object that is visible to the outside | |
of the API through a handle. | |
*/ | |
#include "stdafx.h" | |
#include "adb_api.h" | |
#include "adb_object_handle.h" | |
/// Global ADBAPIHANDLE -> AdbObjectHandle* map | |
AdbObjectHandleMap the_map; | |
/// Locker for the AdbObjectHandleMap instance | |
CComAutoCriticalSection the_map_locker; | |
/// Next adb handle value generator | |
ULONG_PTR next_adb_handle_value = 0; | |
AdbObjectHandle::AdbObjectHandle(AdbObjectType obj_type) | |
: adb_handle_(NULL), | |
object_type_(obj_type), | |
ref_count_(1) { | |
ATLASSERT(obj_type < AdbObjectTypeMax); | |
} | |
AdbObjectHandle::~AdbObjectHandle() { | |
ATLASSERT(0 == ref_count_); | |
ATLASSERT(NULL == adb_handle_); | |
} | |
LONG AdbObjectHandle::AddRef() { | |
ATLASSERT(ref_count_ > 0); | |
return InterlockedIncrement(&ref_count_); | |
} | |
LONG AdbObjectHandle::Release() { | |
ATLASSERT(ref_count_ > 0); | |
LONG ret = InterlockedDecrement(&ref_count_); | |
ATLASSERT(ret >= 0); | |
if (0 == ret) { | |
LastReferenceReleased(); | |
delete this; | |
} | |
return ret; | |
} | |
ADBAPIHANDLE AdbObjectHandle::CreateHandle() { | |
ADBAPIHANDLE ret = NULL; | |
// We have to hold this lock while we're dealing with the handle | |
// and the table | |
the_map_locker.Lock(); | |
ATLASSERT(!IsOpened()); | |
if (!IsOpened()) { | |
try { | |
// Generate next handle value | |
next_adb_handle_value++; | |
ret = reinterpret_cast<ADBAPIHANDLE>(next_adb_handle_value); | |
// Add ourselves to the map | |
the_map[ret] = this; | |
// Save handle, addref and return | |
adb_handle_ = ret; | |
AddRef(); | |
} catch (...) { | |
ret = NULL; | |
SetLastError(ERROR_OUTOFMEMORY); | |
} | |
} else { | |
// Signaling that this object is already opened | |
SetLastError(ERROR_GEN_FAILURE); | |
} | |
the_map_locker.Unlock(); | |
return ret; | |
} | |
bool AdbObjectHandle::CloseHandle() { | |
bool ret = false; | |
// Addref just in case that last reference to this object is being | |
// held in the map | |
AddRef(); | |
the_map_locker.Lock(); | |
ATLASSERT(IsOpened()); | |
if (IsOpened()) { | |
try { | |
// Look us up in the map. | |
AdbObjectHandleMap::iterator found = the_map.find(adb_handle()); | |
ATLASSERT((found != the_map.end()) && (this == found->second)); | |
if ((found != the_map.end()) && (this == found->second)) { | |
// Remove ourselves from the map, close and release the object | |
the_map.erase(found); | |
adb_handle_ = NULL; | |
Release(); | |
ret = true; | |
} else { | |
SetLastError(ERROR_INVALID_HANDLE); | |
} | |
} catch (...) { | |
ret = false; | |
SetLastError(ERROR_OUTOFMEMORY); | |
} | |
} else { | |
SetLastError(ERROR_INVALID_HANDLE); | |
} | |
the_map_locker.Unlock(); | |
Release(); | |
return ret; | |
} | |
bool AdbObjectHandle::IsObjectOfType(AdbObjectType obj_type) const { | |
return (obj_type == object_type()); | |
} | |
void AdbObjectHandle::LastReferenceReleased() { | |
ATLASSERT(!IsOpened()); | |
} | |
AdbObjectHandle* AdbObjectHandle::Lookup(ADBAPIHANDLE adb_hndl) { | |
AdbObjectHandle* ret = NULL; | |
the_map_locker.Lock(); | |
try { | |
// Look us up in the map. | |
AdbObjectHandleMap::iterator found = the_map.find(adb_hndl); | |
if (found != the_map.end()) { | |
ret = found->second; | |
ret->AddRef(); | |
} | |
} catch (...) { | |
SetLastError(ERROR_OUTOFMEMORY); | |
} | |
the_map_locker.Unlock(); | |
return ret; | |
} |