dev: Add support for single-pass scan out in the PixelPump
Add a helper function to scan out an entire frame in one time step. This requires the public PixelPump to be changed somewhat to separate timing updates from general PixelPump control. Instead of calling PixelPump::start(timings), timings now need to be updated using a separate call to PixelPump::updateTimings(timings) before calling PixelPump::start(). Display controllers that don't need accurate timing (e.g., in KVM mode), can use the new PixelPump::renderFrame() API to render an entire frame in one step. This call results in the same callbacks (e.g., calls to nextPixel()) as the timing calls, but they all happen in immediately. Unlike the timing counterpart, renderFrame() doesn't support buffer underruns and will panic if nextPixle() indicates an underrun. Change-Id: I76c84db04249b02d4207c5281d82aa693d0881be Signed-off-by: Andreas Sandberg <andreas.sandberg@arm.com> Reviewed-on: https://gem5-review.googlesource.com/2241 Reviewed-by: Rahul Thakur <rjthakur@google.com> Reviewed-by: Jason Lowe-Power <jason@lowepower.com>
This commit is contained in:
parent
82a8230aa7
commit
9dd54d10ab
3 changed files with 82 additions and 11 deletions
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2010-2013, 2015 ARM Limited
|
* Copyright (c) 2010-2013, 2015, 2017 ARM Limited
|
||||||
* All rights reserved
|
* All rights reserved
|
||||||
*
|
*
|
||||||
* The license below extends only to copyright in the software and shall
|
* The license below extends only to copyright in the software and shall
|
||||||
|
@ -204,8 +204,11 @@ HDLcd::drainResume()
|
||||||
// We restored from an old checkpoint without a pixel pump, start
|
// We restored from an old checkpoint without a pixel pump, start
|
||||||
// an new refresh. This typically happens when restoring from old
|
// an new refresh. This typically happens when restoring from old
|
||||||
// checkpoints.
|
// checkpoints.
|
||||||
if (enabled() && !pixelPump.active())
|
if (enabled() && !pixelPump.active()) {
|
||||||
pixelPump.start(displayTimings());
|
// Update timing parameter before rendering frames
|
||||||
|
pixelPump.updateTimings(displayTimings());
|
||||||
|
pixelPump.start();
|
||||||
|
}
|
||||||
|
|
||||||
// We restored from a checkpoint and need to update the VNC server
|
// We restored from a checkpoint and need to update the VNC server
|
||||||
if (pixelPump.active() && vnc)
|
if (pixelPump.active() && vnc)
|
||||||
|
@ -476,7 +479,10 @@ HDLcd::cmdEnable()
|
||||||
{
|
{
|
||||||
createDmaEngine();
|
createDmaEngine();
|
||||||
conv = pixelConverter();
|
conv = pixelConverter();
|
||||||
pixelPump.start(displayTimings());
|
|
||||||
|
// Update timing parameter before rendering frames
|
||||||
|
pixelPump.updateTimings(displayTimings());
|
||||||
|
pixelPump.start();
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2015 ARM Limited
|
* Copyright (c) 2015, 2017 ARM Limited
|
||||||
* All rights reserved
|
* All rights reserved
|
||||||
*
|
*
|
||||||
* The license below extends only to copyright in the software and shall
|
* The license below extends only to copyright in the software and shall
|
||||||
|
@ -136,10 +136,11 @@ BasePixelPump::unserialize(CheckpointIn &cp)
|
||||||
event->unserializeSection(cp, event->name());
|
event->unserializeSection(cp, event->name());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void
|
void
|
||||||
BasePixelPump::start(const DisplayTimings &timings)
|
BasePixelPump::updateTimings(const DisplayTimings &timings)
|
||||||
{
|
{
|
||||||
|
panic_if(active(), "Trying to update timings in active PixelPump\n");
|
||||||
|
|
||||||
_timings = timings;
|
_timings = timings;
|
||||||
|
|
||||||
// Resize the frame buffer if needed
|
// Resize the frame buffer if needed
|
||||||
|
@ -149,9 +150,15 @@ BasePixelPump::start(const DisplayTimings &timings)
|
||||||
// Set the current line past the last line in the frame. This
|
// Set the current line past the last line in the frame. This
|
||||||
// triggers the new frame logic in beginLine().
|
// triggers the new frame logic in beginLine().
|
||||||
line = _timings.linesPerFrame();
|
line = _timings.linesPerFrame();
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
BasePixelPump::start()
|
||||||
|
{
|
||||||
schedule(evBeginLine, clockEdge());
|
schedule(evBeginLine, clockEdge());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void
|
void
|
||||||
BasePixelPump::stop()
|
BasePixelPump::stop()
|
||||||
{
|
{
|
||||||
|
@ -241,6 +248,54 @@ BasePixelPump::renderPixels()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
BasePixelPump::renderFrame()
|
||||||
|
{
|
||||||
|
_underrun = false;
|
||||||
|
line = 0;
|
||||||
|
|
||||||
|
// Signal vsync end and render the frame
|
||||||
|
line = _timings.lineVBackPorchStart();
|
||||||
|
onVSyncEnd();
|
||||||
|
|
||||||
|
// We only care about the visible screen area when rendering the
|
||||||
|
// frame
|
||||||
|
for (line = _timings.lineFirstVisible();
|
||||||
|
line < _timings.lineFrontPorchStart();
|
||||||
|
++line) {
|
||||||
|
|
||||||
|
_posX = 0;
|
||||||
|
|
||||||
|
onHSyncBegin();
|
||||||
|
onHSyncEnd();
|
||||||
|
|
||||||
|
renderLine();
|
||||||
|
}
|
||||||
|
|
||||||
|
line = _timings.lineFrontPorchStart() - 1;
|
||||||
|
onFrameDone();
|
||||||
|
|
||||||
|
// Signal vsync until the next frame begins
|
||||||
|
line = _timings.lineVSyncStart();
|
||||||
|
onVSyncBegin();
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
BasePixelPump::renderLine()
|
||||||
|
{
|
||||||
|
const unsigned pos_y(posY());
|
||||||
|
|
||||||
|
Pixel pixel(0, 0, 0);
|
||||||
|
for (_posX = 0; _posX < _timings.width; ++_posX) {
|
||||||
|
if (!nextPixel(pixel)) {
|
||||||
|
panic("Unexpected underrun in BasePixelPump (%u, %u)\n",
|
||||||
|
_posX, pos_y);
|
||||||
|
}
|
||||||
|
fb.pixel(_posX, pos_y) = pixel;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
BasePixelPump::PixelEvent::PixelEvent(
|
BasePixelPump::PixelEvent::PixelEvent(
|
||||||
const char *name, BasePixelPump *_parent, CallbackType _func)
|
const char *name, BasePixelPump *_parent, CallbackType _func)
|
||||||
: Event(), Drainable(),
|
: Event(), Drainable(),
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright (c) 2015 ARM Limited
|
* Copyright (c) 2015, 2017 ARM Limited
|
||||||
* All rights reserved
|
* All rights reserved
|
||||||
*
|
*
|
||||||
* The license below extends only to copyright in the software and shall
|
* The license below extends only to copyright in the software and shall
|
||||||
|
@ -148,15 +148,22 @@ class BasePixelPump
|
||||||
public Serializable
|
public Serializable
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
BasePixelPump(EventManager &em, ClockDomain &pxl_clk, unsigned pixel_chunk);
|
BasePixelPump(EventManager &em, ClockDomain &pxl_clk,
|
||||||
|
unsigned pixel_chunk);
|
||||||
virtual ~BasePixelPump();
|
virtual ~BasePixelPump();
|
||||||
|
|
||||||
void serialize(CheckpointOut &cp) const override;
|
void serialize(CheckpointOut &cp) const override;
|
||||||
void unserialize(CheckpointIn &cp) override;
|
void unserialize(CheckpointIn &cp) override;
|
||||||
|
|
||||||
public: // Public API
|
public: // Public API
|
||||||
/** Starting pushing pixels using the supplied display timings. */
|
/** Update frame size using display timing */
|
||||||
void start(const DisplayTimings &timings);
|
void updateTimings(const DisplayTimings &timings);
|
||||||
|
|
||||||
|
/** Render an entire frame in KVM execution mode */
|
||||||
|
void renderFrame();
|
||||||
|
|
||||||
|
/** Starting pushing pixels in timing mode */
|
||||||
|
void start();
|
||||||
|
|
||||||
/** Immediately stop pushing pixels */
|
/** Immediately stop pushing pixels */
|
||||||
void stop();
|
void stop();
|
||||||
|
@ -285,6 +292,9 @@ class BasePixelPump
|
||||||
void beginLine();
|
void beginLine();
|
||||||
void renderPixels();
|
void renderPixels();
|
||||||
|
|
||||||
|
/** Fast and event-free line rendering function */
|
||||||
|
void renderLine();
|
||||||
|
|
||||||
/** Convenience vector when doing operations on all events */
|
/** Convenience vector when doing operations on all events */
|
||||||
std::vector<PixelEvent *> pixelEvents;
|
std::vector<PixelEvent *> pixelEvents;
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue