2016-11-26 23:41:33 -06:00
|
|
|
|
using System;
|
|
|
|
|
using System.Linq;
|
|
|
|
|
using Microsoft.VisualStudio.TestTools.UnitTesting;
|
|
|
|
|
using SpriteCompiler.Problem;
|
|
|
|
|
using SpriteCompiler.AI;
|
2016-11-29 00:20:29 -06:00
|
|
|
|
using System.Diagnostics;
|
2016-11-30 00:43:54 -06:00
|
|
|
|
using System.Collections.Generic;
|
2016-11-26 23:41:33 -06:00
|
|
|
|
|
|
|
|
|
namespace SpriteCompiler.Test
|
|
|
|
|
{
|
|
|
|
|
[TestClass]
|
|
|
|
|
public class Tests
|
|
|
|
|
{
|
|
|
|
|
[TestMethod]
|
|
|
|
|
public void TestEmptySprite()
|
|
|
|
|
{
|
|
|
|
|
// Arrange
|
|
|
|
|
var problem = SpriteGeneratorSearchProblem.CreateSearchProblem();
|
|
|
|
|
var search = SpriteGeneratorSearchProblem.Create();
|
|
|
|
|
|
|
|
|
|
// Act : solve the problem
|
2016-12-08 23:15:59 -06:00
|
|
|
|
var solution = search.Search(problem, SpriteGeneratorState.Init(Enumerable.Empty<SpriteByte>()));
|
2016-11-26 23:41:33 -06:00
|
|
|
|
|
|
|
|
|
// Assert : The initial state IS the goal state
|
|
|
|
|
Assert.AreEqual(1, solution.Count());
|
|
|
|
|
}
|
2016-11-29 00:20:29 -06:00
|
|
|
|
|
|
|
|
|
[TestMethod]
|
|
|
|
|
public void TestSingleByteSprite()
|
|
|
|
|
{
|
2016-11-29 00:51:06 -06:00
|
|
|
|
Trace.WriteLine("Testing a sprite with just one byte");
|
|
|
|
|
|
2016-11-29 00:20:29 -06:00
|
|
|
|
// Arrange
|
|
|
|
|
var problem = SpriteGeneratorSearchProblem.CreateSearchProblem();
|
|
|
|
|
var search = SpriteGeneratorSearchProblem.Create();
|
|
|
|
|
|
|
|
|
|
// Act : solve the problem
|
2016-12-08 21:59:46 -06:00
|
|
|
|
var solution = search.Search(problem, SpriteGeneratorState.Init(new byte[] { 0xAA }));
|
2016-11-29 00:20:29 -06:00
|
|
|
|
|
|
|
|
|
// Assert
|
|
|
|
|
//
|
|
|
|
|
// The fastest way to draw a single byte at the current location should be
|
|
|
|
|
//
|
|
|
|
|
// TCS
|
|
|
|
|
// SHORT A
|
|
|
|
|
// LDA #$AA
|
|
|
|
|
// STA 0,s
|
|
|
|
|
// LONG A = 14 cycles
|
|
|
|
|
|
|
|
|
|
// Write out the solution
|
2016-11-30 00:43:54 -06:00
|
|
|
|
WriteOutSolution(solution);
|
2016-11-30 16:07:22 -06:00
|
|
|
|
|
|
|
|
|
Assert.AreEqual(5, solution.Count());
|
|
|
|
|
Assert.AreEqual(14, (int)solution.Last().PathCost);
|
2016-11-29 00:20:29 -06:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
[TestMethod]
|
|
|
|
|
public void TestSingleWordSprite()
|
|
|
|
|
{
|
|
|
|
|
// Arrange
|
|
|
|
|
var problem = SpriteGeneratorSearchProblem.CreateSearchProblem();
|
|
|
|
|
var search = SpriteGeneratorSearchProblem.Create();
|
|
|
|
|
|
|
|
|
|
// Act : solve the problem
|
2016-12-08 21:59:46 -06:00
|
|
|
|
var solution = search.Search(problem, SpriteGeneratorState.Init(new byte[] { 0xAA, 0x55 }));
|
2016-11-29 00:20:29 -06:00
|
|
|
|
|
2016-11-30 00:43:54 -06:00
|
|
|
|
// Assert
|
|
|
|
|
//
|
|
|
|
|
// The fastest way to draw a single word at the current location should be
|
|
|
|
|
//
|
|
|
|
|
// TCS
|
|
|
|
|
// LDA #$55AA
|
|
|
|
|
// STA 0,s = 10 cycles
|
|
|
|
|
|
2016-11-30 16:07:22 -06:00
|
|
|
|
// Write out the solution
|
|
|
|
|
WriteOutSolution(solution);
|
|
|
|
|
|
2016-11-30 00:43:54 -06:00
|
|
|
|
Assert.AreEqual(3, solution.Count());
|
|
|
|
|
Assert.AreEqual(10, (int)solution.Last().PathCost);
|
2016-11-30 16:07:22 -06:00
|
|
|
|
}
|
|
|
|
|
|
2016-12-05 23:17:53 -06:00
|
|
|
|
[TestMethod]
|
|
|
|
|
public void TestOverlappingWrite()
|
|
|
|
|
{
|
|
|
|
|
// Arrange
|
|
|
|
|
var problem = SpriteGeneratorSearchProblem.CreateSearchProblem();
|
|
|
|
|
var search = SpriteGeneratorSearchProblem.Create();
|
|
|
|
|
|
|
|
|
|
// Act : solve the problem
|
2016-12-08 21:59:46 -06:00
|
|
|
|
var solution = search.Search(problem, SpriteGeneratorState.Init(new byte[] { 0x11, 0x22, 0x22 }));
|
2016-12-05 23:17:53 -06:00
|
|
|
|
|
|
|
|
|
// Assert
|
|
|
|
|
//
|
|
|
|
|
// Solution should be 18 cycles
|
|
|
|
|
//
|
|
|
|
|
// TCS
|
|
|
|
|
// LDA #$2211
|
|
|
|
|
// STA 0,s
|
|
|
|
|
// LDA #$2222
|
|
|
|
|
// STA 1,s = 18 cycles
|
|
|
|
|
|
|
|
|
|
// Write out the solution
|
|
|
|
|
WriteOutSolution(solution);
|
|
|
|
|
|
|
|
|
|
Assert.AreEqual(18, (int)solution.Last().PathCost);
|
|
|
|
|
}
|
|
|
|
|
|
2016-12-08 23:15:59 -06:00
|
|
|
|
[TestMethod]
|
|
|
|
|
public void TestSingleByteWithMask()
|
|
|
|
|
{
|
|
|
|
|
// Arrange
|
|
|
|
|
var problem = SpriteGeneratorSearchProblem.CreateSearchProblem();
|
|
|
|
|
var search = SpriteGeneratorSearchProblem.Create();
|
|
|
|
|
|
|
|
|
|
// Act : solve the problem
|
|
|
|
|
var data = new byte[] { 0x00, 0x11 };
|
|
|
|
|
var mask = new byte[] { 0xFF, 0x00 };
|
|
|
|
|
|
|
|
|
|
var solution = search.Search(problem, SpriteGeneratorState.Init(data, mask));
|
|
|
|
|
|
|
|
|
|
// Assert
|
|
|
|
|
//
|
|
|
|
|
// Solution should be same as regular single-byte sprite, except
|
|
|
|
|
// the store happens at offset 1, instead of 0.
|
|
|
|
|
|
|
|
|
|
// Write out the solution
|
|
|
|
|
WriteOutSolution(solution);
|
|
|
|
|
|
|
|
|
|
Assert.AreEqual(5, solution.Count());
|
|
|
|
|
Assert.AreEqual(14, (int)solution.Last().PathCost);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
[TestMethod]
|
|
|
|
|
public void TestSinglePixelMask()
|
|
|
|
|
{
|
|
|
|
|
// Arrange
|
|
|
|
|
var problem = SpriteGeneratorSearchProblem.CreateSearchProblem();
|
|
|
|
|
var search = SpriteGeneratorSearchProblem.Create();
|
|
|
|
|
|
|
|
|
|
// Act : solve the problem
|
|
|
|
|
var data = new byte[] { 0x01, 0x11 };
|
|
|
|
|
var mask = new byte[] { 0xF0, 0x00 };
|
|
|
|
|
|
|
|
|
|
var solution = search.Search(problem, SpriteGeneratorState.Init(data, mask));
|
|
|
|
|
|
|
|
|
|
// Assert
|
|
|
|
|
//
|
|
|
|
|
// Solution should be a single 16-bit RMW
|
|
|
|
|
//
|
|
|
|
|
// TCS
|
|
|
|
|
// LDA 0,s
|
|
|
|
|
// AND #$00F0
|
|
|
|
|
// ORA #$1101
|
|
|
|
|
// STA 0,s = 18 cycles
|
|
|
|
|
|
|
|
|
|
// Write out the solution
|
|
|
|
|
WriteOutSolution(solution);
|
|
|
|
|
|
|
|
|
|
Assert.AreEqual(18, (int)solution.Last().PathCost);
|
|
|
|
|
}
|
|
|
|
|
|
2016-11-30 16:07:22 -06:00
|
|
|
|
[TestMethod]
|
|
|
|
|
public void TestConsecutiveWordSprite()
|
|
|
|
|
{
|
|
|
|
|
// Arrange
|
|
|
|
|
var problem = SpriteGeneratorSearchProblem.CreateSearchProblem();
|
|
|
|
|
var search = SpriteGeneratorSearchProblem.Create();
|
|
|
|
|
|
|
|
|
|
// Act : solve the problem
|
2016-12-08 21:59:46 -06:00
|
|
|
|
var solution = search.Search(problem, SpriteGeneratorState.Init(new byte[] { 0xAA, 0x55, 0x11, 0x22, 0x33, 0x44, 0x55, 0x66 }));
|
2016-11-30 16:07:22 -06:00
|
|
|
|
|
|
|
|
|
// Assert
|
|
|
|
|
//
|
2016-12-08 23:30:31 -06:00
|
|
|
|
// The fastest way to draw a consecutive words should be
|
2016-11-30 16:07:22 -06:00
|
|
|
|
//
|
|
|
|
|
// ADC #7
|
|
|
|
|
// TCS
|
|
|
|
|
// PEA $6655
|
|
|
|
|
// PEA $4433
|
|
|
|
|
// PEA $2211
|
|
|
|
|
// PEA $55AA = 25 cycles
|
2016-11-30 00:43:54 -06:00
|
|
|
|
|
|
|
|
|
// Write out the solution
|
|
|
|
|
WriteOutSolution(solution);
|
2016-11-30 16:07:22 -06:00
|
|
|
|
|
2016-12-05 21:55:04 -06:00
|
|
|
|
Assert.AreEqual(6, solution.Count());
|
|
|
|
|
Assert.AreEqual(25, (int)solution.Last().PathCost);
|
2016-11-30 00:43:54 -06:00
|
|
|
|
}
|
|
|
|
|
|
2016-12-08 23:30:31 -06:00
|
|
|
|
[TestMethod]
|
|
|
|
|
public void TestConsecutiveWordSpriteWithMask()
|
|
|
|
|
{
|
|
|
|
|
// Arrange
|
|
|
|
|
var problem = SpriteGeneratorSearchProblem.CreateSearchProblem();
|
|
|
|
|
var search = SpriteGeneratorSearchProblem.Create();
|
|
|
|
|
|
|
|
|
|
// Act : solve the problem
|
|
|
|
|
var data = new byte[] { 0x01, 0x11, 0x22, 0x11, 0x33 };
|
|
|
|
|
var mask = new byte[] { 0xF0, 0x00, 0x00, 0x00, 0x00 };
|
|
|
|
|
|
|
|
|
|
var solution = search.Search(problem, SpriteGeneratorState.Init(data, mask));
|
|
|
|
|
|
|
|
|
|
// Assert
|
|
|
|
|
//
|
|
|
|
|
// The fastest way to render this data should be
|
|
|
|
|
//
|
|
|
|
|
// ADC #4
|
|
|
|
|
// TCS
|
|
|
|
|
// PEA $3311
|
|
|
|
|
// PEA $2211
|
|
|
|
|
// LDA 0,s
|
|
|
|
|
// AND #$00F0
|
|
|
|
|
// ORA #$1101
|
|
|
|
|
// STA 0,s = 31 cycles
|
|
|
|
|
|
|
|
|
|
// Write out the solution
|
|
|
|
|
WriteOutSolution(solution);
|
|
|
|
|
|
|
|
|
|
Assert.AreEqual(31, (int)solution.Last().PathCost);
|
|
|
|
|
}
|
|
|
|
|
|
2016-11-30 00:43:54 -06:00
|
|
|
|
private void WriteOutSolution(IEnumerable<SpriteGeneratorSearchNode> solution)
|
|
|
|
|
{
|
|
|
|
|
foreach (var step in solution.Skip(1))
|
|
|
|
|
{
|
|
|
|
|
Trace.WriteLine(step.Action.ToString());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Trace.WriteLine(string.Format("; Total Cost = {0} cycles", (int)solution.Last().PathCost));
|
2016-11-29 00:20:29 -06:00
|
|
|
|
}
|
2016-11-26 23:41:33 -06:00
|
|
|
|
}
|
|
|
|
|
}
|