VBoxDbgBase.cpp revision a33af978add1a03aab11b2895f441af5cb2a11a6
/* $Id$ */
/** @file
* VBox Debugger GUI - Base classes.
*/
/*
* Copyright (C) 2006-2007 Oracle Corporation
*
* Oracle Corporation confidential
* All rights reserved
*/
/*******************************************************************************
* Header Files *
*******************************************************************************/
#define LOG_GROUP LOG_GROUP_DBGG
#include <VBox/err.h>
#include <iprt/asm.h>
#include <iprt/assert.h>
#include <limits.h>
#include "VBoxDbgBase.h"
#include "VBoxDbgGui.h"
#include <QApplication>
#include <QWidgetList>
VBoxDbgBase::VBoxDbgBase(VBoxDbgGui *a_pDbgGui)
: m_pDbgGui(a_pDbgGui), m_pVM(NULL), m_hGUIThread(RTThreadNativeSelf())
{
/*
* Register
*/
PVM pVM = a_pDbgGui->getVMHandle();
if (pVM)
{
m_pVM = pVM;
int rc = VMR3AtStateRegister(pVM, atStateChange, this);
AssertRC(rc);
}
}
VBoxDbgBase::~VBoxDbgBase()
{
/*
* If the VM is still around.
*/
/** @todo need to do some locking here? */
PVM pVM = ASMAtomicXchgPtrT(&m_pVM, NULL, PVM);
if (pVM)
{
int rc = VMR3AtStateDeregister(pVM, atStateChange, this);
AssertRC(rc);
}
}
int
VBoxDbgBase::stamReset(const QString &rPat)
{
QByteArray Utf8Array = rPat.toUtf8();
const char *pszPat = !rPat.isEmpty() ? Utf8Array.constData() : NULL;
PVM pVM = m_pVM;
if ( pVM
&& VMR3GetState(pVM) < VMSTATE_DESTROYING)
return STAMR3Reset(pVM, pszPat);
return VERR_INVALID_HANDLE;
}
int
VBoxDbgBase::stamEnum(const QString &rPat, PFNSTAMR3ENUM pfnEnum, void *pvUser)
{
QByteArray Utf8Array = rPat.toUtf8();
const char *pszPat = !rPat.isEmpty() ? Utf8Array.constData() : NULL;
PVM pVM = m_pVM;
if ( pVM
&& VMR3GetState(pVM) < VMSTATE_DESTROYING)
return STAMR3Enum(pVM, pszPat, pfnEnum, pvUser);
return VERR_INVALID_HANDLE;
}
int
VBoxDbgBase::dbgcCreate(PDBGCBACK pBack, unsigned fFlags)
{
PVM pVM = m_pVM;
if ( pVM
&& VMR3GetState(pVM) < VMSTATE_DESTROYING)
return DBGCCreate(pVM, pBack, fFlags);
return VERR_INVALID_HANDLE;
}
/*static*/ DECLCALLBACK(void)
VBoxDbgBase::atStateChange(PVM pVM, VMSTATE enmState, VMSTATE /*enmOldState*/, void *pvUser)
{
VBoxDbgBase *pThis = (VBoxDbgBase *)pvUser;
switch (enmState)
{
case VMSTATE_TERMINATED:
/** @todo need to do some locking here? */
if (ASMAtomicCmpXchgPtr(&pThis->m_pVM, NULL, pVM))
pThis->sigTerminated();
break;
case VMSTATE_DESTROYING:
pThis->sigDestroying();
break;
default:
break;
}
}
void
VBoxDbgBase::sigDestroying()
{
}
void
VBoxDbgBase::sigTerminated()
{
}
//
//
//
// V B o x D b g B a s e W i n d o w
// V B o x D b g B a s e W i n d o w
// V B o x D b g B a s e W i n d o w
//
//
//
unsigned VBoxDbgBaseWindow::m_cxBorder = 0;
unsigned VBoxDbgBaseWindow::m_cyBorder = 0;
VBoxDbgBaseWindow::VBoxDbgBaseWindow(VBoxDbgGui *a_pDbgGui, QWidget *a_pParent)
: QWidget(a_pParent, Qt::Window), VBoxDbgBase(a_pDbgGui),
#ifdef Q_WS_X11
m_fPolished(false),
#else
m_fPolished(true),
#endif
m_x(INT_MAX), m_y(INT_MAX), m_cx(0), m_cy(0)
{
}
VBoxDbgBaseWindow::~VBoxDbgBaseWindow()
{
}
void
VBoxDbgBaseWindow::vShow()
{
show();
/** @todo this ain't working right. HELP! */
setWindowState(windowState() & ~Qt::WindowMinimized);
//activateWindow();
//setFocus();
vPolishSizeAndPos();
}
void
VBoxDbgBaseWindow::vReposition(int a_x, int a_y, unsigned a_cx, unsigned a_cy, bool a_fResize)
{
if (a_fResize)
{
m_cx = a_cx;
m_cy = a_cy;
QSize BorderSize = frameSize() - size();
if (BorderSize == QSize(0,0))
BorderSize = vGuessBorderSizes();
resize(a_cx - BorderSize.width(), a_cy - BorderSize.height());
}
m_x = a_x;
m_y = a_y;
move(a_x, a_y);
}
bool
VBoxDbgBaseWindow::event(QEvent *a_pEvt)
{
bool fRc = QWidget::event(a_pEvt);
vPolishSizeAndPos();
return fRc;
}
void
VBoxDbgBaseWindow::vPolishSizeAndPos()
{
/* Ignore if already done or no size set. */
if ( m_fPolished
|| (m_x == INT_MAX && m_y == INT_MAX))
return;
QSize BorderSize = frameSize() - size();
if (BorderSize != QSize(0,0))
m_fPolished = true;
vReposition(m_x, m_y, m_cx, m_cy, m_cx || m_cy);
}
QSize
VBoxDbgBaseWindow::vGuessBorderSizes()
{
#ifdef Q_WS_X11 /* (from the qt gui) */
/* only once. */
if (!m_cxBorder && !m_cyBorder)
{
/* On X11, there is no way to determine frame geometry (including WM
* decorations) before the widget is shown for the first time. Stupidly
* enumerate other top level widgets to find the thickest frame. The code
* is based on the idea taken from QDialog::adjustPositionInternal(). */
int extraw = 0, extrah = 0;
QWidgetList list = QApplication::topLevelWidgets();
QListIterator<QWidget*> it (list);
while ((extraw == 0 || extrah == 0) && it.hasNext())
{
int framew, frameh;
QWidget *current = it.next();
if (!current->isVisible())
continue;
framew = current->frameGeometry().width() - current->width();
frameh = current->frameGeometry().height() - current->height();
extraw = qMax (extraw, framew);
extrah = qMax (extrah, frameh);
}
if (extraw || extrah)
{
m_cxBorder = extraw;
m_cyBorder = extrah;
}
}
#endif /* X11 */
return QSize(m_cxBorder, m_cyBorder);
}