/*
* Copyright (C) 2013 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.
*/
#include "jdwp/jdwp.h"
#include "base/stringprintf.h"
#include "jdwp/jdwp_priv.h"
namespace art {
namespace JDWP {
Request::Request(const uint8_t* bytes, uint32_t available) : p_(bytes) {
byte_count_ = Read4BE();
end_ = bytes + byte_count_;
CHECK_LE(byte_count_, available);
id_ = Read4BE();
int8_t flags = Read1();
if ((flags & kJDWPFlagReply) != 0) {
LOG(FATAL) << "reply?!";
}
command_set_ = Read1();
command_ = Read1();
}
Request::~Request() {
}
void Request::CheckConsumed() {
if (p_ < end_) {
CHECK(p_ == end_) << "read too few bytes: " << (end_ - p_);
} else if (p_ > end_) {
CHECK(p_ == end_) << "read too many bytes: " << (p_ - end_);
}
}
std::string Request::ReadUtf8String() {
uint32_t length = Read4BE();
std::string s;
s.resize(length);
memcpy(&s[0], p_, length);
p_ += length;
VLOG(jdwp) << " string \"" << s << "\"";
return s;
}
// Helper function: read a variable-width value from the input buffer.
uint64_t Request::ReadValue(size_t width) {
uint64_t value = -1;
switch (width) {
case 1: value = Read1(); break;
case 2: value = Read2BE(); break;
case 4: value = Read4BE(); break;
case 8: value = Read8BE(); break;
default: LOG(FATAL) << width; break;
}
return value;
}
int32_t Request::ReadSigned32(const char* what) {
int32_t value = static_cast<int32_t>(Read4BE());
VLOG(jdwp) << " " << what << " " << value;
return value;
}
uint32_t Request::ReadUnsigned32(const char* what) {
uint32_t value = Read4BE();
VLOG(jdwp) << " " << what << " " << value;
return value;
}
FieldId Request::ReadFieldId() {
FieldId id = Read4BE();
VLOG(jdwp) << " field id " << DescribeField(id);
return id;
}
MethodId Request::ReadMethodId() {
MethodId id = Read4BE();
VLOG(jdwp) << " method id " << DescribeMethod(id);
return id;
}
ObjectId Request::ReadObjectId(const char* specific_kind) {
ObjectId id = Read8BE();
VLOG(jdwp) << StringPrintf(" %s id %#llx", specific_kind, id);
return id;
}
ObjectId Request::ReadArrayId() {
return ReadObjectId("array");
}
ObjectId Request::ReadObjectId() {
return ReadObjectId("object");
}
ObjectId Request::ReadThreadId() {
return ReadObjectId("thread");
}
ObjectId Request::ReadThreadGroupId() {
return ReadObjectId("thread group");
}
RefTypeId Request::ReadRefTypeId() {
RefTypeId id = Read8BE();
VLOG(jdwp) << " ref type id " << DescribeRefTypeId(id);
return id;
}
FrameId Request::ReadFrameId() {
FrameId id = Read8BE();
VLOG(jdwp) << " frame id " << id;
return id;
}
JdwpTag Request::ReadTag() {
return ReadEnum1<JdwpTag>("tag");
}
JdwpTypeTag Request::ReadTypeTag() {
return ReadEnum1<JdwpTypeTag>("type tag");
}
JdwpLocation Request::ReadLocation() {
JdwpLocation location;
memset(&location, 0, sizeof(location)); // Allows memcmp(3) later.
location.type_tag = ReadTypeTag();
location.class_id = ReadObjectId("class");
location.method_id = ReadMethodId();
location.dex_pc = Read8BE();
VLOG(jdwp) << " location " << location;
return location;
}
JdwpModKind Request::ReadModKind() {
return ReadEnum1<JdwpModKind>("mod kind");
}
uint8_t Request::Read1() {
return *p_++;
}
uint16_t Request::Read2BE() {
uint16_t result = p_[0] << 8 | p_[1];
p_ += 2;
return result;
}
uint32_t Request::Read4BE() {
uint32_t result = p_[0] << 24;
result |= p_[1] << 16;
result |= p_[2] << 8;
result |= p_[3];
p_ += 4;
return result;
}
uint64_t Request::Read8BE() {
uint64_t high = Read4BE();
uint64_t low = Read4BE();
return (high << 32) | low;
}
} // namespace JDWP
} // namespace art