// Copyright (c) 2013 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 "ppapi/proxy/serialized_handle.h"
#include "base/memory/shared_memory.h"
#include "base/pickle.h"
#include "base/platform_file.h"
#include "build/build_config.h"
#include "ipc/ipc_platform_file.h"
#if defined(OS_NACL)
#include <unistd.h>
#endif
namespace ppapi {
namespace proxy {
SerializedHandle::SerializedHandle()
: type_(INVALID),
shm_handle_(base::SharedMemory::NULLHandle()),
size_(0),
descriptor_(IPC::InvalidPlatformFileForTransit()) {
}
SerializedHandle::SerializedHandle(Type type_param)
: type_(type_param),
shm_handle_(base::SharedMemory::NULLHandle()),
size_(0),
descriptor_(IPC::InvalidPlatformFileForTransit()) {
}
SerializedHandle::SerializedHandle(const base::SharedMemoryHandle& handle,
uint32 size)
: type_(SHARED_MEMORY),
shm_handle_(handle),
size_(size),
descriptor_(IPC::InvalidPlatformFileForTransit()) {
}
SerializedHandle::SerializedHandle(
Type type,
const IPC::PlatformFileForTransit& socket_descriptor)
: type_(type),
shm_handle_(base::SharedMemory::NULLHandle()),
size_(0),
descriptor_(socket_descriptor) {
}
bool SerializedHandle::IsHandleValid() const {
switch (type_) {
case SHARED_MEMORY:
return base::SharedMemory::IsHandleValid(shm_handle_);
case SOCKET:
case CHANNEL_HANDLE:
case FILE:
return !(IPC::InvalidPlatformFileForTransit() == descriptor_);
case INVALID:
return false;
// No default so the compiler will warn us if a new type is added.
}
return false;
}
void SerializedHandle::Close() {
if (IsHandleValid()) {
switch (type_) {
case INVALID:
NOTREACHED();
break;
case SHARED_MEMORY:
base::SharedMemory::CloseHandle(shm_handle_);
break;
case SOCKET:
case CHANNEL_HANDLE:
case FILE:
base::PlatformFile file =
IPC::PlatformFileForTransitToPlatformFile(descriptor_);
#if !defined(OS_NACL)
base::ClosePlatformFile(file);
#else
close(file);
#endif
break;
// No default so the compiler will warn us if a new type is added.
}
}
*this = SerializedHandle();
}
// static
bool SerializedHandle::WriteHeader(const Header& hdr, Pickle* pickle) {
if (!pickle->WriteInt(hdr.type))
return false;
if (hdr.type == SHARED_MEMORY) {
if (!pickle->WriteUInt32(hdr.size))
return false;
}
if (hdr.type == FILE) {
if (!pickle->WriteInt(hdr.open_flag))
return false;
}
return true;
}
// static
bool SerializedHandle::ReadHeader(PickleIterator* iter, Header* hdr) {
*hdr = Header(INVALID, 0, 0);
int type = 0;
if (!iter->ReadInt(&type))
return false;
bool valid_type = false;
switch (type) {
case SHARED_MEMORY: {
uint32 size = 0;
if (!iter->ReadUInt32(&size))
return false;
hdr->size = size;
valid_type = true;
break;
}
case FILE: {
int open_flag = 0;
if (!iter->ReadInt(&open_flag))
return false;
hdr->open_flag = open_flag;
valid_type = true;
}
case SOCKET:
case CHANNEL_HANDLE:
case INVALID:
valid_type = true;
break;
// No default so the compiler will warn us if a new type is added.
}
if (valid_type)
hdr->type = Type(type);
return valid_type;
}
} // namespace proxy
} // namespace ppapi