/*
VdpTMS9918ACanvasRenderer.java
(c) 2011-2014 Edward Swartz
All rights reserved. This program and the accompanying materials
are made available under the terms of the Eclipse Public License v1.0
which accompanies this distribution, and is available at
http://www.eclipse.org/legal/epl-v10.html
*/
package v9t9.video.tms9918a;
import static v9t9.common.hardware.VdpTMS9918AConsts.REG_ST;
import java.util.Arrays;
import v9t9.common.client.ISettingsHandler;
import v9t9.common.client.IVideoRenderer;
import v9t9.common.video.RedrawBlock;
import v9t9.video.IVdpModeBlockRedrawHandler;
import v9t9.video.IVdpModeRedrawHandler;
/**
* This is a renderer for the TI-99/4A VDP chip which renders to an IVdpCanvas
* by tracking 8x8 or 6x8 blocks that need to be redrawn.
*
* @author ejs
*/
public class VdpTMS9918ACanvasBlockRenderer extends BaseVdpTMS9918ACanvasRenderer {
private RedrawBlock[] blocks;
protected IVdpModeBlockRedrawHandler vdpModeRedrawHandler;
public VdpTMS9918ACanvasBlockRenderer(ISettingsHandler settings, IVideoRenderer renderer) {
super(settings, renderer);
}
/* (non-Javadoc)
* @see v9t9.video.tms9918a.BaseVdpTMS9918ACanvasRenderer#setModeRedrawHandler(v9t9.video.IVdpModeRedrawHandler)
*/
@Override
protected void setModeRedrawHandler(IVdpModeRedrawHandler redrawHandler) {
this.vdpModeRedrawHandler = (IVdpModeBlockRedrawHandler) redrawHandler;
}
/** preinitialize the update blocks with the sizes for this mode */
protected void initUpdateBlocks(int blockWidth) {
int w = blockWidth;
int h = 8;
if (blocks == null) {
blocks = new RedrawBlock[getMaxRedrawblocks()];
for (int i = 0; i < blocks.length; i++) {
blocks[i] = new RedrawBlock();
}
}
if (blocks[0].w != blockWidth) {
for (int i = 0; i < blocks.length; i++) {
blocks[i].w = w;
blocks[i].h = h;
}
}
}
protected final void setVideoMode() {
}
/* (non-Javadoc)
* @see v9t9.video.tms9918a.BaseVdpTMS9918ACanvasRenderer#setBitmapMode()
*/
@Override
protected void setBitmapMode() {
super.setBitmapMode();
initUpdateBlocks(8);
}
/* (non-Javadoc)
* @see v9t9.video.tms9918a.BaseVdpTMS9918ACanvasRenderer#setBlankMode()
*/
@Override
protected void setBlankMode() {
super.setBlankMode();
initUpdateBlocks(8);
if (vdpModeRedrawHandler != null)
vdpModeRedrawHandler.clear();
}
/* (non-Javadoc)
* @see v9t9.video.tms9918a.BaseVdpTMS9918ACanvasRenderer#setGraphicsMode()
*/
@Override
protected void setGraphicsMode() {
super.setGraphicsMode();
initUpdateBlocks(8);
}
/* (non-Javadoc)
* @see v9t9.video.tms9918a.BaseVdpTMS9918ACanvasRenderer#setMultiMode()
*/
@Override
protected void setMultiMode() {
super.setMultiMode();
initUpdateBlocks(8);
}
/* (non-Javadoc)
* @see v9t9.video.tms9918a.BaseVdpTMS9918ACanvasRenderer#setTextMode()
*/
@Override
protected void setTextMode() {
super.setTextMode();
initUpdateBlocks(6);
}
protected int getMaxRedrawblocks() {
return 1024;
}
public synchronized boolean update() {
flushVdpChanges(vdpModeRedrawHandler);
if (!vdpChanges.changed)
return false;
//System.out.println(System.currentTimeMillis());
if (vdpModeRedrawHandler != null) {
//long start = System.currentTimeMillis();
int count = 0;
// don't let video rendering happen in middle of updating
synchronized (vdpCanvas) {
if (colorsChanged) {
vdpCanvas.syncColors();
colorsChanged = false;
}
vdpModeRedrawHandler.prepareUpdate();
if (vdpChanges.fullRedraw) {
// clear for the actual mode (not blank mode)
vdpModeRedrawHandler.clear();
vdpCanvas.markDirty();
}
if (!isBlank()) {
if (spriteRedrawHandler != null && drawSprites) {
byte vdpStatus = (byte) vdpChip.getRegister(REG_ST);
vdpStatus = spriteRedrawHandler.updateSpriteCoverage(
vdpStatus, vdpChanges.fullRedraw);
if (!pauseMachine.getBoolean())
vdpChip.setRegister(REG_ST, vdpStatus);
}
if (vdpChanges.fullRedraw) {
vdpChanges.screen.set(0, getMaxRedrawblocks());
}
count = vdpModeRedrawHandler.updateCanvas(blocks);
if (spriteRedrawHandler != null && drawSprites) {
spriteRedrawHandler.updateCanvas(vdpChanges.fullRedraw);
}
}
vdpCanvas.markDirty(blocks, count);
vdpChanges.screen.clear();
Arrays.fill(vdpChanges.patt, (byte) 0);
Arrays.fill(vdpChanges.color, (byte) 0);
if (drawSprites) {
Arrays.fill(vdpChanges.sprpat, (byte) 0);
vdpChanges.sprite = 0;
}
vdpChanges.fullRedraw = false;
vdpChanges.changed = false;
}
//System.out.println("elapsed: " + (System.currentTimeMillis() - start));
}
return true;
}
}