/*
* Copyright (c) 2010, 2012, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
*
* This code is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License version 2 only, as
* published by the Free Software Foundation.
*
* This code is distributed in the hope that it will be useful, but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
* version 2 for more details (a copy is included in the LICENSE file that
* accompanied this code).
*
* You should have received a copy of the GNU General Public License version
* 2 along with this work; if not, write to the Free Software Foundation,
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
*
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
* or visit www.oracle.com if you need additional information or have any
* questions.
*/
package com.sun.max.tele.debug.unix;
import java.io.*;
import java.net.*;
import java.nio.*;
import com.sun.max.lang.*;
import com.sun.max.program.*;
import com.sun.max.tele.*;
import com.sun.max.tele.channel.*;
import com.sun.max.tele.channel.natives.*;
import com.sun.max.tele.debug.*;
import com.sun.max.tele.util.*;
import com.sun.max.unsafe.*;
import com.sun.max.vm.hosted.*;
/**
* Provides default implementations of the methods in {@link TeleChannelDataIOProtocol}.
*
* Makes use of {@link TeleChannelNatives} to invoke the corresponding native code.
*
*/
public class UnixNativeTeleChannelProtocolAdaptor implements TeleChannelProtocol {
private long processHandle;
private TeleChannelNatives natives;
private boolean bigEndian;
protected TeleVMAgent agent;
public UnixNativeTeleChannelProtocolAdaptor(TeleChannelNatives natives) {
if (natives == null) {
natives = new TeleChannelNatives();
}
this.natives = natives;
}
@Override
public boolean initialize(int tlaSize, boolean bigEndian) {
this.bigEndian = bigEndian;
natives.teleInitialize(tlaSize);
return true;
}
protected long createBufferAndAgent(String programFile, String[] commandLineArguments) throws BootImageException {
final long commandLineArgumentsBuffer = TeleProcess.createCommandLineArgumentsBuffer(new File(programFile), commandLineArguments);
this.agent = new TeleVMAgent();
agent.start();
return commandLineArgumentsBuffer;
}
@Override
public long create(String programFile, String[] commandLineArguments) {
final long commandLineArgumentsBuffer;
try {
commandLineArgumentsBuffer = createBufferAndAgent(programFile, commandLineArguments);
} catch (BootImageException ex) {
return -1;
}
processHandle = natives.createChild(commandLineArgumentsBuffer, agent.port());
return processHandle;
}
@Override
public boolean attach(int id) {
return false;
}
@Override
public boolean detach() {
return false;
}
@Override
public long getBootHeapStart() {
try {
final Socket socket = agent.waitForVM();
final InputStream stream = socket.getInputStream();
final Endianness endianness = bigEndian ? Endianness.BIG : Endianness.LITTLE;
final Pointer heap = Word.read(stream, endianness).asPointer();
Trace.line(1, "Received boot image address from VM: 0x" + heap.toHexString());
socket.close();
agent.close();
return heap.toLong();
} catch (Exception ioException) {
TeleError.unexpected("Error while reading boot image address from VM process", ioException);
return 0;
}
}
@Override
public int maxByteBufferSize() {
// no limit
return Integer.MAX_VALUE;
}
@Override
public boolean readRegisters(long threadId, byte[] integerRegisters, int integerRegistersSize, byte[] floatingPointRegisters, int floatingPointRegistersSize, byte[] stateRegisters,
int stateRegistersSize) {
return natives.readRegisters(processHandle, threadId,
integerRegisters, integerRegisters.length,
floatingPointRegisters, floatingPointRegisters.length,
stateRegisters, stateRegisters.length);
}
@Override
public boolean setInstructionPointer(long threadId, long ip) {
return natives.setInstructionPointer(processHandle, threadId, ip);
}
@Override
public boolean singleStep(long threadId) {
return natives.singleStep(processHandle, threadId);
}
@Override
public int waitUntilStoppedAsInt() {
return natives.waitUntilStopped(processHandle);
}
@Override
public ProcessState waitUntilStopped() {
final int state = waitUntilStoppedAsInt();
return ProcessState.values()[state];
}
@Override
public boolean suspendAll() {
return natives.suspend(processHandle);
}
@Override
public boolean resumeAll() {
return natives.resume(processHandle);
}
@Override
public boolean suspend(long threadId) {
return natives.suspend(processHandle, threadId);
}
@Override
public boolean resume(long threadId) {
return natives.resume(threadId);
}
@Override
public boolean kill() {
natives.kill(processHandle);
return true;
}
@Override
public boolean activateWatchpoint(long start, long size, boolean after, boolean read, boolean write, boolean exec) {
return natives.activateWatchpoint(processHandle, start, size, after, read, write, exec);
}
@Override
public boolean deactivateWatchpoint(long start, long size) {
return natives.deactivateWatchpoint(processHandle, start, size);
}
@Override
public long readWatchpointAddress() {
return natives.readWatchpointAddress(processHandle);
}
@Override
public int readWatchpointAccessCode() {
return natives.readWatchpointAccessCode(processHandle);
}
@Override
public int setTransportDebugLevel(int level) {
return 0;
}
@Override
public int readBytes(long src, byte[] dst, int dstOffset, int length) {
return natives.readBytes(processHandle, src, dst, false, dstOffset, length);
}
@Override
public int writeBytes(long dst, byte[] src, int srcOffset, int length) {
return natives.writeBytes(processHandle, dst, src, false, srcOffset, length);
}
@Override
public int readBytes(long src, ByteBuffer dst, int dstOffset, int length) {
if (dst.isDirect()) {
return natives.readBytes(processHandle, src, dst, true, dstOffset, length);
}
return natives.readBytes(processHandle, src, dst.array(), false, dst.arrayOffset() + dstOffset, length);
}
@Override
public int writeBytes(long dst, ByteBuffer src, int srcOffset, int length) {
if (src.isDirect()) {
return natives.writeBytes(processHandle, dst, src, true, srcOffset, length);
}
return natives.writeBytes(processHandle, dst, src.array(), false, src.arrayOffset() + srcOffset, length);
}
@Override
public boolean gatherThreads(Object teleDomain, Object threadList, long tlaList) {
natives.gatherThreads(processHandle, teleDomain, threadList, tlaList);
return true;
}
@Override
public int gatherThreads(long tlaList) {
TeleError.unexpected("TeleChannelProtocol.gatherThreads(int, int) should not be called in this configuration");
return 0;
}
@Override
public int readThreads(int size, byte[] gatherThreadsData) {
TeleError.unexpected("TeleChannelProtocol.readThreads should not be called in this configuration");
return 0;
}
}