1
0
mirror of https://github.com/2martens/uni.git synced 2026-05-06 19:36:26 +02:00
Files
uni/es/blatt6/uebung6-1/uebung6-1.ino
Jim Martens b1733ead11 [ES] Finished 6.1
Signed-off-by: Jim Martens <github@2martens.de>
2015-07-07 16:28:01 +02:00

488 lines
13 KiB
C++

#include <SPI.h>
#include <SD.h>
// pins
int slaveSelectPin = 10;
int SDslaveSelectPin = 4;
int rstPin = 6;
int dcPin = 5;
// constants
int divider = 84;
int maxBufferIndex = 504;
// buffer
char screenBuffer[504];
// ASCII data
unsigned char font[95][6] =
{
{ 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 }, // space
{ 0x00, 0x00, 0x5F, 0x00, 0x00, 0x00 }, // !
{ 0x00, 0x07, 0x00, 0x07, 0x00, 0x00 }, // "
{ 0x14, 0x7F, 0x14, 0x7F, 0x14, 0x00 }, // #
{ 0x24, 0x2A, 0x7F, 0x2A, 0x12, 0x00 }, // $
{ 0x23, 0x13, 0x08, 0x64, 0x62, 0x00 }, // %
{ 0x36, 0x49, 0x55, 0x22, 0x50, 0x00 }, // &
{ 0x00, 0x00, 0x07, 0x00, 0x00, 0x00 }, // '
{ 0x00, 0x1C, 0x22, 0x41, 0x00, 0x00 }, // (
{ 0x00, 0x41, 0x22, 0x1C, 0x00, 0x00 }, // )
{ 0x0A, 0x04, 0x1F, 0x04, 0x0A, 0x00 }, // *
{ 0x08, 0x08, 0x3E, 0x08, 0x08, 0x00 }, // +
{ 0x00, 0x50, 0x30, 0x00, 0x00, 0x00 }, // ,
{ 0x08, 0x08, 0x08, 0x08, 0x08, 0x00 }, // -
{ 0x00, 0x60, 0x60, 0x00, 0x00, 0x00 }, // .
{ 0x20, 0x10, 0x08, 0x04, 0x02, 0x00 }, // slash
{ 0x3E, 0x51, 0x49, 0x45, 0x3E, 0x00 }, // 0
{ 0x00, 0x42, 0x7F, 0x40, 0x00, 0x00 }, // 1
{ 0x42, 0x61, 0x51, 0x49, 0x46, 0x00 }, // 2
{ 0x21, 0x41, 0x45, 0x4B, 0x31, 0x00 }, // 3
{ 0x18, 0x14, 0x12, 0x7F, 0x10, 0x00 }, // 4
{ 0x27, 0x45, 0x45, 0x45, 0x39, 0x00 }, // 5
{ 0x3C, 0x4A, 0x49, 0x49, 0x30, 0x00 }, // 6
{ 0x03, 0x71, 0x09, 0x05, 0x03, 0x00 }, // 7
{ 0x36, 0x49, 0x49, 0x49, 0x36, 0x00 }, // 8
{ 0x06, 0x49, 0x49, 0x29, 0x1E, 0x00 }, // 9
{ 0x00, 0x36, 0x36, 0x00, 0x00, 0x00 }, // :
{ 0x00, 0x56, 0x36, 0x00, 0x00, 0x00 }, // ;
{ 0x08, 0x14, 0x22, 0x41, 0x00, 0x00 }, // <
{ 0x14, 0x14, 0x14, 0x14, 0x14, 0x00 }, // =
{ 0x00, 0x41, 0x22, 0x14, 0x08, 0x00 }, // >
{ 0x02, 0x01, 0x51, 0x09, 0x06, 0x00 }, // ?
{ 0x32, 0x49, 0x79, 0x41, 0x3E, 0x00 }, // @
{ 0x7E, 0x11, 0x11, 0x11, 0x7E, 0x00 }, // A
{ 0x7F, 0x49, 0x49, 0x49, 0x36, 0x00 }, // B
{ 0x3E, 0x41, 0x41, 0x41, 0x22, 0x00 }, // C
{ 0x7F, 0x41, 0x41, 0x41, 0x3E, 0x00 }, // D
{ 0x7F, 0x49, 0x49, 0x49, 0x41, 0x00 }, // E
{ 0x7F, 0x09, 0x09, 0x09, 0x01, 0x00 }, // F
{ 0x3E, 0x41, 0x41, 0x49, 0x7A, 0x00 }, // G
{ 0x7F, 0x08, 0x08, 0x08, 0x7F, 0x00 }, // H
{ 0x00, 0x41, 0x7F, 0x41, 0x00, 0x00 }, // I
{ 0x20, 0x40, 0x41, 0x3F, 0x01, 0x00 }, // J
{ 0x7F, 0x08, 0x14, 0x22, 0x41, 0x00 }, // K
{ 0x7F, 0x40, 0x40, 0x40, 0x40, 0x00 }, // L
{ 0x7F, 0x02, 0x0C, 0x02, 0x7F, 0x00 }, // M
{ 0x7F, 0x04, 0x08, 0x10, 0x7F, 0x00 }, // N
{ 0x3E, 0x41, 0x41, 0x41, 0x3E, 0x00 }, // O
{ 0x7F, 0x09, 0x09, 0x09, 0x06, 0x00 }, // P
{ 0x3E, 0x41, 0x51, 0x21, 0x5E, 0x00 }, // Q
{ 0x7F, 0x09, 0x19, 0x29, 0x46, 0x00 }, // R
{ 0x26, 0x49, 0x49, 0x49, 0x32, 0x00 }, // S
{ 0x01, 0x01, 0x7F, 0x01, 0x01, 0x00 }, // T
{ 0x3F, 0x40, 0x40, 0x40, 0x3F, 0x00 }, // U
{ 0x1F, 0x20, 0x40, 0x20, 0x1F, 0x00 }, // V
{ 0x3F, 0x40, 0x38, 0x40, 0x3F, 0x00 }, // W
{ 0x63, 0x14, 0x08, 0x14, 0x63, 0x00 }, // X
{ 0x07, 0x08, 0x70, 0x08, 0x07, 0x00 }, // Y
{ 0x61, 0x51, 0x49, 0x45, 0x43, 0x00 }, // Z
{ 0x00, 0x7F, 0x41, 0x41, 0x00, 0x00 }, // [
{ 0x02, 0x04, 0x08, 0x10, 0x20, 0x00 }, // backslash
{ 0x00, 0x41, 0x41, 0x7F, 0x00, 0x00 }, // ]
{ 0x04, 0x02, 0x01, 0x02, 0x04, 0x00 }, // ^
{ 0x40, 0x40, 0x40, 0x40, 0x40, 0x00 }, // _
{ 0x00, 0x01, 0x02, 0x04, 0x00, 0x00 }, // `
{ 0x20, 0x54, 0x54, 0x54, 0x78, 0x00 }, // a
{ 0x7F, 0x48, 0x44, 0x44, 0x38, 0x00 }, // b
{ 0x38, 0x44, 0x44, 0x44, 0x20, 0x00 }, // c
{ 0x38, 0x44, 0x44, 0x48, 0x7F, 0x00 }, // d
{ 0x38, 0x54, 0x54, 0x54, 0x18, 0x00 }, // e
{ 0x08, 0x7E, 0x09, 0x01, 0x02, 0x00 }, // f
{ 0x08, 0x54, 0x54, 0x54, 0x3C, 0x00 }, // g
{ 0x7F, 0x08, 0x04, 0x04, 0x78, 0x00 }, // h
{ 0x00, 0x48, 0x7D, 0x40, 0x00, 0x00 }, // i
{ 0x20, 0x40, 0x44, 0x3D, 0x00, 0x00 }, // j
{ 0x7F, 0x10, 0x28, 0x44, 0x00, 0x00 }, // k
{ 0x00, 0x41, 0x7F, 0x40, 0x00, 0x00 }, // l
{ 0x7C, 0x04, 0x78, 0x04, 0x78, 0x00 }, // m
{ 0x7C, 0x08, 0x04, 0x04, 0x78, 0x00 }, // n
{ 0x38, 0x44, 0x44, 0x44, 0x38, 0x00 }, // o
{ 0x7C, 0x14, 0x14, 0x14, 0x08, 0x00 }, // p
{ 0x08, 0x14, 0x14, 0x18, 0x7C, 0x00 }, // q
{ 0x7C, 0x08, 0x04, 0x04, 0x08, 0x00 }, // r
{ 0x48, 0x54, 0x54, 0x54, 0x20, 0x00 }, // s
{ 0x04, 0x3F, 0x44, 0x40, 0x20, 0x00 }, // t
{ 0x3C, 0x40, 0x40, 0x20, 0x7C, 0x00 }, // u
{ 0x1C, 0x20, 0x40, 0x20, 0x1C, 0x00 }, // v
{ 0x3C, 0x40, 0x30, 0x40, 0x3C, 0x00 }, // w
{ 0x44, 0x28, 0x10, 0x28, 0x44, 0x00 }, // x
{ 0x0C, 0x50, 0x50, 0x50, 0x3C, 0x00 }, // y
{ 0x44, 0x64, 0x54, 0x4C, 0x44, 0x00 }, // z
{ 0x00, 0x08, 0x36, 0x41, 0x00, 0x00 }, // {
{ 0x00, 0x00, 0x7F, 0x00, 0x00, 0x00 }, // |
{ 0x00, 0x41, 0x36, 0x08, 0x00, 0x00 }, // }
{ 0x10, 0x08, 0x08, 0x10, 0x08, 0x00 } // ~
};
// tmp vars
int readStringLength = 0;
void setup()
{
Serial.begin(9600);
// initialize SPI
SPI.begin(slaveSelectPin);
SPI.setClockDivider(slaveSelectPin, divider);
// initialize SD
SD.begin(SDslaveSelectPin);
// initialize pins
pinMode(dcPin, OUTPUT);
pinMode(rstPin, OUTPUT);
// initial reset of display
resetDisplay();
// initialization of display
digitalWrite(dcPin, LOW);
SPI.transfer(slaveSelectPin, 0x21);
SPI.transfer(slaveSelectPin, 0x14);
SPI.transfer(slaveSelectPin, 0xb0);
SPI.transfer(slaveSelectPin, 0x20);
SPI.transfer(slaveSelectPin, 0x0c);
digitalWrite(dcPin, HIGH);
// initialize screen buffer
resetBuffer();
}
/**
* Loop function for main code
*/
void loop()
{
if (Serial.available() > 0) {
String command = Serial.readString();
resetBuffer();
handleCommand(command);
flushBuffer();
delay(500);
}
}
/**
* Resets display.
*/
void resetDisplay()
{
digitalWrite(rstPin, LOW);
delay(500);
digitalWrite(rstPin, HIGH);
}
/**
* Writes the screen buffer into the display.
*
* It does NOT reset the buffer.
*/
void flushBuffer()
{
for (int i = 0; i < maxBufferIndex; i++) {
SPI.transfer(slaveSelectPin, screenBuffer[i]);
}
}
/**
* Resets the buffer.
*
* Writes 0 in all fields.
*/
void resetBuffer()
{
for (int i = 0; i < maxBufferIndex; i++) {
screenBuffer[i] = 0x0;
}
}
/**
* Sets the pixel at the given location.
* @param x
* @param y
* @param value
*/
void setPixel(int x, int y, int value)
{
int bank = x / 8;
int relativeRow = x - bank * 8;
int bankStartIndex = bank * 84;
int index = bankStartIndex + y;
char bitValue = 0x0;
if (value == 1) {
bitValue = 0xff;
}
char existingValue = screenBuffer[index];
char pixelMask = 1 << relativeRow;
char invMask = ~ pixelMask;
char newValue = pixelMask & bitValue;
char finalExistingValue = existingValue & invMask;
char finalValue = finalExistingValue | newValue;
screenBuffer[index] = finalValue;
}
/**
* Prints given char at given location.
*
* @param x
* @param y
* @param value
*
* @return 0 on success, -1 otherwise
*/
int printChar(int x, int y, char value)
{
// guardians
if (x > 40 || y > 78) {
// absolute guardian
return -1;
}
// TODO relative guardian to existing buffer
char space = ' ';
int arrayIndex = value - space;
for (int i = 0; i < 6; i++) {
char data = font[arrayIndex][i];
for (int _x = 0; _x < 8; _x++) {
char bitmask = 1 << _x;
char pixel = bitmask & data;
int pixelInt = 0;
if (pixel != 0x0) {
pixelInt = 1;
}
setPixel(x + _x, y + i, pixelInt);
}
}
return 0;
}
/**
* Prints a string.
*
* @param int x
* @param int y
* @param char* text
* @param int textLength
*/
int printString(int x, int y, const char* text, int textLength)
{
// guardians
if (x > 40 || y > 78) {
// absolute guardian
return -1;
}
int actualY = y;
int actualX = x;
bool writeError = false;
for (int i = 0; i < textLength; i++) {
char character = text[i];
int result = 0;
// write dash
if ((actualY + 6) > 72) {
printChar(actualX, actualY, '-');
actualX += 10;
actualY = y;
result = printChar(actualX, actualY, character);
}
else {
result = printChar(actualX, actualY, character);
}
actualY += 6;
if (result != 0) {
// the character would be out of the display x bounds
writeError = true;
break;
}
}
if (writeError) {
return 0;
resetBuffer();
printString(3, 20, "Too long", 8);
}
return 0;
}
/**
* Prints an image with given parameters.
*
* @param int x
* @param int y
* @param int rows
* @param int cols
* @param const int* pixels
*/
void printImage(int x, int y, int rows, int cols, const int* pixels)
{
int currentX = x;
int currentY = y;
int maxIndex = rows * cols;
for (int i = 0; i < maxIndex; i++) {
if (currentY == (y + cols)) {
currentX++;
currentY = y;
}
int pixel = pixels[i];
setPixel(currentX, currentY, pixel);
currentY++;
}
}
/**
* Reads the contents of a text file into a string.
*
* @param File file
* @param char* text
*/
void readString(File file, char* text, unsigned long fileSize)
{
if (file) {
int readBytes = file.readBytesUntil('\n', text, fileSize);
text[readBytes] = '\0';
readStringLength = readBytes;
}
}
/**
* Explodes the string into an integer array.
*
* @param const int* input
* @param int length
* @param char delimeter
* @param int* result
*/
void explodeString(const char* input, int length, char delimeter, int* result)
{
char tmp[length];
int nthChar = 0;
int nthInt = 0;
for (int i = 0; i < length; i++) {
char currentChar = input[i];
if (currentChar != delimeter) {
tmp[nthChar] = currentChar;
nthChar++;
}
else {
char *end;
tmp[nthChar] = '\0';
result[nthInt] = (int) strtol(tmp, &end, 10);
nthChar = 0;
memset(&tmp[0], 0, length);
nthInt++;
}
}
char *end;
tmp[nthChar] = '\0';
result[nthInt] = (int) strtol(tmp, &end, 10);
nthChar = 0;
memset(&tmp[0], 0, length);
nthInt++;
}
/**
* Handles the command.
*
* @param String command
*/
void handleCommand(String command)
{
String existCommand = "doesFileExist";
String showCommand = "showFile";
// determine command and parameter
int indexP1 = command.indexOf('(');
int indexP2 = command.indexOf(')');
String command2 = command;
command2.remove(indexP1);
if (command2 != existCommand && command2 != showCommand) {
Serial.println("Use doesFileExist(<file>) or showFile(<file>)");
return;
}
command.remove(indexP2);
String parameter = command.substring(indexP1 + 1);
char paramChar[parameter.length() + 1];
parameter.toCharArray(paramChar, parameter.length() + 1);
paramChar[parameter.length()] = '\0';
if (command2 == existCommand) {
bool fileExists = SD.exists(paramChar);
Serial.print("Does file exist? ");
Serial.println(fileExists ? "Yes" : "No");
}
if (command2 == showCommand) {
showFile(parameter);
}
}
/**
* Shows the given file.
*
* @param String fileName
*/
void showFile(String fileName)
{
bool isTextFile = false;
int extPos = fileName.indexOf('.');
String ext = fileName.substring(extPos + 1);
isTextFile = (ext == "txt");
char fileNameChar[fileName.length() + 1];
fileName.toCharArray(fileNameChar, fileName.length() + 1);
fileNameChar[fileName.length()] = '\0';
if (SD.exists(fileNameChar)) {
File file = SD.open(fileNameChar);
unsigned long fileSize = file.size();
if (isTextFile) {
char text[fileSize];
readString(file, text, fileSize);
Serial.println(text);
printString(0, 0, text, readStringLength);
}
else {
char dimensions[fileSize];
readString(file, dimensions, fileSize);
int dimensionsLength = readStringLength;
char pixels[fileSize];
readString(file, pixels, fileSize);
int pixelsLength = readStringLength;
// parse dimensions and pixels
int dimensionsInt[2];
explodeString(dimensions, dimensionsLength, ',', dimensionsInt);
int pixelsInt[pixelsLength];
explodeString(pixels, pixelsLength, ',', pixelsInt);
// calculate x and y start positions
int rows = dimensionsInt[0];
int cols = dimensionsInt[1];
int x = (48 - rows) / 2;
int y = (84 - cols) / 2;
Serial.println(x);
Serial.println(y);
Serial.println(rows);
Serial.println(cols);
printImage(x, y, rows, cols, pixelsInt);
}
file.close();
}
}