2005-11-18 15:26:20 +03:00
|
|
|
/*
|
2006-02-02 23:19:29 +03:00
|
|
|
* Copyright (c) 2001-2006, Haiku, Inc.
|
2005-11-18 15:26:20 +03:00
|
|
|
* Distributed under the terms of the MIT license.
|
|
|
|
*
|
|
|
|
* Authors:
|
|
|
|
* Adi Oanca <adioanca@myrealbox.com>
|
|
|
|
* Axel Dörfler, axeld@pinc-software.de
|
|
|
|
* Stephan Aßmus, <superstippi@gmx.de>
|
|
|
|
*/
|
|
|
|
|
2005-05-26 13:21:51 +04:00
|
|
|
|
2004-01-12 01:12:55 +03:00
|
|
|
#include <Accelerant.h>
|
|
|
|
#include <Point.h>
|
|
|
|
#include <GraphicsDefs.h>
|
2005-05-26 13:21:51 +04:00
|
|
|
|
|
|
|
#include <stdlib.h>
|
2004-01-12 01:12:55 +03:00
|
|
|
#include <stdio.h>
|
|
|
|
|
2005-11-04 18:54:16 +03:00
|
|
|
#include "DrawingEngine.h"
|
2005-06-24 03:46:17 +04:00
|
|
|
#include "HWInterface.h"
|
|
|
|
|
2004-01-12 01:12:55 +03:00
|
|
|
#include "ServerScreen.h"
|
|
|
|
|
2005-05-26 13:21:51 +04:00
|
|
|
|
2006-03-01 19:08:28 +03:00
|
|
|
static float
|
|
|
|
get_mode_frequency(const display_mode& mode)
|
|
|
|
{
|
|
|
|
// Taken from Screen preferences
|
|
|
|
float timing = float(mode.timing.h_total * mode.timing.v_total);
|
|
|
|
if (timing == 0.0f)
|
|
|
|
return 0.0f;
|
|
|
|
|
|
|
|
return rint(10 * float(mode.timing.pixel_clock * 1000)
|
|
|
|
/ timing) / 10.0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// #pragma mark -
|
|
|
|
|
|
|
|
|
2005-11-18 15:26:20 +03:00
|
|
|
Screen::Screen(::HWInterface *interface, int32 id)
|
2005-06-24 03:46:17 +04:00
|
|
|
: fID(id),
|
2005-11-04 18:23:54 +03:00
|
|
|
fDriver(interface ? new DrawingEngine(interface) : NULL),
|
2006-02-06 16:36:46 +03:00
|
|
|
fHWInterface(interface),
|
|
|
|
fIsDefault(true)
|
2005-06-24 03:46:17 +04:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2005-11-18 15:26:20 +03:00
|
|
|
|
2005-06-24 03:46:17 +04:00
|
|
|
Screen::Screen()
|
|
|
|
: fID(-1),
|
|
|
|
fDriver(NULL),
|
2006-02-06 16:36:46 +03:00
|
|
|
fHWInterface(NULL),
|
|
|
|
fIsDefault(true)
|
2004-01-12 01:12:55 +03:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2005-05-26 13:21:51 +04:00
|
|
|
|
2005-06-24 03:46:17 +04:00
|
|
|
Screen::~Screen()
|
2004-02-12 04:25:41 +03:00
|
|
|
{
|
2005-06-24 03:46:17 +04:00
|
|
|
Shutdown();
|
2005-06-08 05:40:22 +04:00
|
|
|
delete fDriver;
|
2005-06-24 03:46:17 +04:00
|
|
|
delete fHWInterface;
|
2004-01-12 01:12:55 +03:00
|
|
|
}
|
|
|
|
|
2005-05-26 13:21:51 +04:00
|
|
|
|
2005-06-24 03:46:17 +04:00
|
|
|
status_t
|
|
|
|
Screen::Initialize()
|
2004-02-12 04:25:41 +03:00
|
|
|
{
|
2006-11-29 12:27:23 +03:00
|
|
|
if (fHWInterface) {
|
|
|
|
// init the graphics hardware
|
|
|
|
return fHWInterface->Initialize();
|
2005-06-24 03:46:17 +04:00
|
|
|
}
|
2006-02-06 16:36:46 +03:00
|
|
|
|
|
|
|
return B_NO_INIT;
|
2005-06-24 03:46:17 +04:00
|
|
|
}
|
2004-01-12 01:12:55 +03:00
|
|
|
|
2005-05-26 13:21:51 +04:00
|
|
|
|
2005-06-24 03:46:17 +04:00
|
|
|
void
|
|
|
|
Screen::Shutdown()
|
|
|
|
{
|
2006-11-29 12:27:23 +03:00
|
|
|
if (fHWInterface)
|
|
|
|
fHWInterface->Shutdown();
|
2005-06-24 03:46:17 +04:00
|
|
|
}
|
2004-01-12 01:12:55 +03:00
|
|
|
|
|
|
|
|
2005-06-24 03:46:17 +04:00
|
|
|
status_t
|
2006-02-06 16:36:46 +03:00
|
|
|
Screen::SetMode(display_mode mode, bool makeDefault)
|
2005-06-24 03:46:17 +04:00
|
|
|
{
|
2006-02-06 16:36:46 +03:00
|
|
|
status_t status = fHWInterface->SetMode(mode);
|
2006-11-29 12:27:23 +03:00
|
|
|
// any attached DrawingEngines will be notified
|
2005-06-24 03:46:17 +04:00
|
|
|
|
2006-11-29 12:27:23 +03:00
|
|
|
if (status >= B_OK)
|
2006-02-06 16:36:46 +03:00
|
|
|
fIsDefault = makeDefault;
|
2005-06-24 03:46:17 +04:00
|
|
|
|
2006-02-06 16:36:46 +03:00
|
|
|
return status;
|
2004-01-12 01:12:55 +03:00
|
|
|
}
|
|
|
|
|
2005-05-26 13:21:51 +04:00
|
|
|
|
2005-06-24 03:46:17 +04:00
|
|
|
status_t
|
|
|
|
Screen::SetMode(uint16 width, uint16 height, uint32 colorspace,
|
2006-02-06 16:36:46 +03:00
|
|
|
float frequency, bool makeDefault)
|
2004-02-12 04:25:41 +03:00
|
|
|
{
|
2005-06-24 03:46:17 +04:00
|
|
|
// search for a matching mode
|
2005-05-26 13:21:51 +04:00
|
|
|
display_mode mode;
|
2006-02-06 16:36:46 +03:00
|
|
|
status_t status = _FindMode(width, height, colorspace, frequency, &mode);
|
|
|
|
if (status < B_OK) {
|
2005-07-17 03:43:30 +04:00
|
|
|
// TODO: Move fallback elsewhere, this function should simply
|
|
|
|
// fail if requested to set unsupported mode.
|
2005-06-24 03:46:17 +04:00
|
|
|
// Ups. Not good. Ignore the requested mode and use fallback params.
|
2006-02-06 16:36:46 +03:00
|
|
|
status = _FindMode(640, 480, B_CMAP8, 60.0, &mode);
|
2005-06-24 03:46:17 +04:00
|
|
|
}
|
|
|
|
|
2006-03-01 19:08:28 +03:00
|
|
|
if (status >= B_OK) {
|
|
|
|
float modeFrequency = get_mode_frequency(mode);
|
|
|
|
display_mode originalMode = mode;
|
|
|
|
bool adjusted = false;
|
|
|
|
|
|
|
|
if (modeFrequency != frequency) {
|
|
|
|
// adjust timing to fit the requested frequency if needed
|
|
|
|
// (taken from Screen preferences application)
|
|
|
|
mode.timing.pixel_clock = ((uint32)mode.timing.h_total
|
|
|
|
* mode.timing.v_total / 10 * int32(frequency * 10)) / 1000;
|
|
|
|
adjusted = true;
|
|
|
|
}
|
|
|
|
|
2006-02-06 16:36:46 +03:00
|
|
|
status = SetMode(mode, makeDefault);
|
2006-03-01 19:08:28 +03:00
|
|
|
if (status < B_OK) {
|
|
|
|
// try again with the unchanged mode
|
|
|
|
status = SetMode(originalMode, makeDefault);
|
|
|
|
}
|
|
|
|
}
|
2005-06-24 03:46:17 +04:00
|
|
|
|
2006-02-06 16:36:46 +03:00
|
|
|
return status;
|
2005-06-24 03:46:17 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
Screen::GetMode(display_mode* mode) const
|
|
|
|
{
|
|
|
|
fHWInterface->GetMode(mode);
|
2004-01-12 01:12:55 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2005-05-26 13:21:51 +04:00
|
|
|
void
|
2005-06-24 03:46:17 +04:00
|
|
|
Screen::GetMode(uint16 &width, uint16 &height, uint32 &colorspace,
|
|
|
|
float &frequency) const
|
2005-05-26 13:21:51 +04:00
|
|
|
{
|
|
|
|
display_mode mode;
|
2005-06-24 03:46:17 +04:00
|
|
|
fHWInterface->GetMode(&mode);
|
2005-05-26 13:21:51 +04:00
|
|
|
|
|
|
|
width = mode.virtual_width;
|
|
|
|
height = mode.virtual_height;
|
|
|
|
colorspace = mode.space;
|
2006-03-01 19:08:28 +03:00
|
|
|
frequency = get_mode_frequency(mode);
|
2004-01-12 01:12:55 +03:00
|
|
|
}
|
|
|
|
|
2005-05-26 13:21:51 +04:00
|
|
|
|
2005-07-15 16:45:23 +04:00
|
|
|
BRect
|
|
|
|
Screen::Frame() const
|
|
|
|
{
|
|
|
|
display_mode mode;
|
|
|
|
fHWInterface->GetMode(&mode);
|
|
|
|
|
|
|
|
return BRect(0, 0, mode.virtual_width - 1, mode.virtual_height - 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2005-11-26 19:55:23 +03:00
|
|
|
color_space
|
|
|
|
Screen::ColorSpace() const
|
|
|
|
{
|
|
|
|
display_mode mode;
|
|
|
|
fHWInterface->GetMode(&mode);
|
|
|
|
|
|
|
|
return (color_space)mode.space;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2005-06-24 03:46:17 +04:00
|
|
|
status_t
|
|
|
|
Screen::_FindMode(uint16 width, uint16 height, uint32 colorspace,
|
2005-07-17 02:50:17 +04:00
|
|
|
float frequency, display_mode* mode) const
|
2004-02-12 04:25:41 +03:00
|
|
|
{
|
2006-03-01 19:08:28 +03:00
|
|
|
display_mode* modes = NULL;
|
2005-06-24 03:46:17 +04:00
|
|
|
uint32 count;
|
|
|
|
|
2006-03-01 19:08:28 +03:00
|
|
|
status_t status = fHWInterface->GetModeList(&modes, &count);
|
2006-02-06 16:36:46 +03:00
|
|
|
if (status < B_OK || count <= 0) {
|
2005-06-24 03:46:17 +04:00
|
|
|
// We've run into quite a problem here! This is a function which is a requirement
|
|
|
|
// for a graphics module. The best thing that we can hope for is 640x480x8 without
|
|
|
|
// knowing anything else. While even this seems like insanity to assume that we
|
|
|
|
// can support this, the only lower mode supported is 640x400, but we shouldn't even
|
|
|
|
// bother with such a pathetic possibility.
|
|
|
|
if (width == 640 && height == 480 && colorspace == B_CMAP8)
|
|
|
|
return B_OK;
|
|
|
|
|
2006-02-06 16:36:46 +03:00
|
|
|
return status;
|
2005-06-24 03:46:17 +04:00
|
|
|
}
|
|
|
|
|
2006-03-01 19:08:28 +03:00
|
|
|
int32 index = _FindMode(modes, count, width, height, colorspace, frequency);
|
2005-07-17 02:50:17 +04:00
|
|
|
if (index < 0) {
|
2006-03-01 19:08:28 +03:00
|
|
|
fprintf(stderr, "mode not found (%d, %d, %f) -> using first mode from list!\n",
|
|
|
|
width, height, frequency);
|
2005-07-17 02:50:17 +04:00
|
|
|
// fallback to first mode from list - TODO: ?!?
|
|
|
|
// NOTE: count > 0 is checked above
|
2006-03-01 19:08:28 +03:00
|
|
|
*mode = modes[0];
|
2005-07-17 02:50:17 +04:00
|
|
|
} else {
|
2006-03-01 19:08:28 +03:00
|
|
|
*mode = modes[index];
|
2005-07-17 02:50:17 +04:00
|
|
|
}
|
|
|
|
|
2006-03-01 19:08:28 +03:00
|
|
|
delete[] modes;
|
2005-07-17 02:50:17 +04:00
|
|
|
|
|
|
|
return B_OK;
|
|
|
|
}
|
|
|
|
|
2005-11-18 15:26:20 +03:00
|
|
|
|
2006-03-01 19:08:28 +03:00
|
|
|
/*!
|
|
|
|
\brief Returns the mode that matches the given criteria. The frequency
|
|
|
|
doesn't have to match, though - the closest one found is used.
|
|
|
|
*/
|
2005-07-17 02:50:17 +04:00
|
|
|
int32
|
2006-03-01 19:08:28 +03:00
|
|
|
Screen::_FindMode(const display_mode* modes, uint32 count,
|
|
|
|
uint16 width, uint16 height, uint32 colorspace,
|
|
|
|
float frequency) const
|
2005-07-17 02:50:17 +04:00
|
|
|
{
|
2006-03-01 19:08:28 +03:00
|
|
|
// we always try to choose the mode with the closest frequency
|
|
|
|
float bestFrequencyDiff = 0.0f;
|
2005-07-17 02:50:17 +04:00
|
|
|
int32 index = -1;
|
2006-03-01 19:08:28 +03:00
|
|
|
|
2005-06-24 03:46:17 +04:00
|
|
|
for (uint32 i = 0; i < count; i++) {
|
2006-03-01 19:08:28 +03:00
|
|
|
if (modes[i].virtual_width == width
|
|
|
|
&& modes[i].virtual_height == height
|
|
|
|
&& modes[i].space == colorspace) {
|
|
|
|
// we have found a mode with the correct width, height and format
|
|
|
|
// now see if the frequency matches
|
|
|
|
float modeFrequency = get_mode_frequency(modes[i]);
|
|
|
|
float frequencyDiff = fabs(modeFrequency - frequency);
|
|
|
|
|
|
|
|
if (index == -1 || bestFrequencyDiff > frequencyDiff) {
|
2005-07-17 02:50:17 +04:00
|
|
|
index = i;
|
2006-03-01 19:08:28 +03:00
|
|
|
if (frequencyDiff == 0.0f)
|
|
|
|
break;
|
|
|
|
|
|
|
|
bestFrequencyDiff = frequencyDiff;
|
2005-06-24 03:46:17 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2006-03-01 19:08:28 +03:00
|
|
|
|
2005-07-17 02:50:17 +04:00
|
|
|
return index;
|
2004-01-12 01:12:55 +03:00
|
|
|
}
|
|
|
|
|