]> git.sur5r.net Git - c128-kasse/blobdiff - src/kasse.c
Initialize memory
[c128-kasse] / src / kasse.c
index 0d95aea56f8ac51a841faab06a2ef6db682cec07..91b4a4a7d832498ce899444282c18f6cf536ffc4 100644 (file)
@@ -1,6 +1,6 @@
 /* 
  * RGB2R-C128-Kassenprogramm
- * (c) 2007 phil_fry, sECuRE, sur5r
+ * (c) 2007-2008 phil_fry, sECuRE, sur5r
  * See LICENSE for license information
  *
  */
 // drucker 4 oder 5
 // graphic 4,0,10
 
+/* NOTE: undocumented function which scratches files
+   We need to use this function because linking unistd.h
+   makes our program break at runtime.
+ */
+unsigned char __fastcall__ _sysremove(const char *name);
+
+static void sane_exit() {
+       save_items();
+       save_credits();
+       exit(1);
+}
 
 /* Hauptbildschirm ausgeben */
 static void print_screen() {
@@ -27,7 +38,7 @@ static void print_screen() {
        char profit[10];
        clrscr();
        if (format_euro(profit, 10, money) == NULL) {
-               cprintf("Einname %ld konnte nicht umgerechnet werden\r\n", money);
+               cprintf("Einnahme %ld konnte nicht umgerechnet werden\r\n", money);
                exit(1);
        }
        cprintf("C128-Kassenprogramm (phil_fry, sECuRE, sur5r)\r\
@@ -39,27 +50,47 @@ Eingenommen: %s, Verkauft: %ld Flaschen, Drucken: %s\r\n\r\n",
                        i, status.status[i].item_name, status.status[i].price, status.status[i].times_sold);
        cprintf("\r\nBefehle: s) Daten sichern d) Drucken umschalten\r\
 g) Guthabenverwaltung     z) Zeit setzen\r\
-q) Beenden\r\n");
+f) Freitext verkaufen     q) Beenden\r\n");
 }
 
 static void log_file(const char *s) {
+       /* A log-entry has usually 50 bytes, so we take 64 bytes.
+          Because files are wrapped (log.0, log.1, ...) every 100
+          lines, we don't need more than 100 * 64 bytes. */
+       char *buffer = malloc(sizeof(char) * 64 * 100);
        char filename[8];
-       BYTE c;
-       sprintf(filename, "log%4d", log_num++);
+       char read = 0;
+       unsigned int c;
+       if (buffer == NULL) {
+               cprintf("No memory available\n");
+               }
+       buffer[0] = '\0';
+       if (((++log_lines_written) % 100) == 0)
+               log_num++;
+       sprintf(filename, "log-%d", log_num);
+       /* Don't read log if there were no lines written before */
+       if (log_lines_written != 1) {
+               if ((c = cbm_open((BYTE)8, (BYTE)8, (BYTE)0, filename)) != 0) {
+                       c128_perror(c, "cbm_open(log)");
+                       sane_exit();
+               }
+               read = cbm_read((BYTE)8, buffer, sizeof(char) * 64 * 100);
+               cbm_close((BYTE)8);
+               _sysremove(filename);
+       }
        if ((c = cbm_open((BYTE)8, (BYTE)8, (BYTE)1, filename)) != 0) {
                c128_perror(c, "cbm_open(log)");
-               save_items();
-               save_credits();
-               exit(1);
+               sane_exit();
        }
-       c = cbm_write((BYTE)8, s, strlen(s));
-       if (c != strlen(s)) {
-               cprintf("Could not save logfile, please make sure the floppy is not full!\n");
-               save_items();
-               save_credits();
-               exit(1);
+       /* TODO: read  < 0? */
+       strcpy(buffer+read, s);
+       c = cbm_write((BYTE)8, buffer, read+strlen(s));
+       if (c != (read+strlen(s))) {
+               cprintf("Could not save logfile (wrote %d bytes, wanted %d bytes), please make sure the floppy is not full!\n", c, (read+strlen(s)));
+               sane_exit();
        }
        cbm_close((BYTE)8);
+       free(buffer);
 }
 
 static char retry_or_quit() {
@@ -71,8 +102,8 @@ static char retry_or_quit() {
        return *c;
 }
 
-/* Druckt eine entsprechende Zeile aus */
-static void print_log(BYTE n, int einheiten, char *nickname) {
+/* Prints a line and logs it to file */
+static void print_log(char *name, int item_price, int einheiten, char *nickname) {
        BYTE c;
        char *time = get_time();
        char price[10];
@@ -84,23 +115,21 @@ static void print_log(BYTE n, int einheiten, char *nickname) {
           Anzahl
           Nickname (falls es vom Guthaben abgezogen wird)
           */
-       if (format_euro(price, 10, status.status[n].price) == NULL) {
-               cprintf("Preis %d konnte nicht umgerechnet werden\r\n", status.status[n].price);
+       if (format_euro(price, 10, item_price) == NULL) {
+               cprintf("Preis %d konnte nicht umgerechnet werden\r\n", item_price);
                exit(1);
        }
                
-       sprintf(print_buffer, "%c[%lu] %s - %s - %s - %d - an %s\r\n",  17,
-                       items_sold, time, status.status[n].item_name, price, 
+       sprintf(print_buffer, "%c[%lu] %s - %s - %s - %d - an %s\r",  17,
+                       items_sold, time, name, price, 
                        einheiten, (*nickname != '\0' ? nickname : "Unbekannt"));
-       RETRY:;
+RETRY:
        c = cbm_open((BYTE)4, (BYTE)4, (BYTE)0, NULL);
        if (c != 0) {
                c128_perror(c, "cbm_open(printer)");
-               if (retry_or_quit() == 'q') {
-                       save_items();
-                       save_credits();
-                       exit(1);
-               }
+               if (retry_or_quit() == 'q')
+                       sane_exit();
+
                goto RETRY;
        }
        c = cbm_write((BYTE)4, print_buffer, strlen(print_buffer));
@@ -117,8 +146,8 @@ static void print_log(BYTE n, int einheiten, char *nickname) {
        log_file(print_buffer);
 }
 
-/* Dialog, der einen durch's Abrechnen der Einträge führt */
-void buy(BYTE n) {
+/* dialog which is called for each bought item */
+BYTE buy(char *name, unsigned int price) {
        int negative = 1;
        char entered[5] = {'1', 0, 0, 0, 0};
        BYTE i = 0, matches = 0;
@@ -128,11 +157,7 @@ void buy(BYTE n) {
        char nickname[11];
        struct credits_t *credit;
 
-       if (status.status[n].item_name == NULL) {
-               cprintf("FEHLER: Diese Einheit existiert nicht.\r\n");
-               return;
-       }
-       cprintf("Wieviel Einheiten \"%s\"? [1] \r\n", status.status[n].item_name);
+       cprintf("Wieviel Einheiten \"%s\"? [1] \r\n", name);
        while (1) {
                c = getchar();
                if (c == 13)
@@ -140,7 +165,7 @@ void buy(BYTE n) {
                else if (c == 27) {
                        cprintf("Kauf abgebrochen, druecke RETURN...\r\n");
                        get_input();
-                       return;
+                       return 1;
                } else if (c == '-' && i == 0)
                        negative = -1;
                else if (c > 47 && c < 58)
@@ -149,7 +174,7 @@ void buy(BYTE n) {
        einheiten = atoi(entered) * negative;
        
        toggle_videomode();
-       cprintf("%dx %s fuer ", einheiten, status.status[n].item_name);
+       cprintf("%dx %s fuer ", einheiten, name);
        toggle_videomode();
        
        cprintf("\r\nAuf ein Guthaben kaufen? Wenn ja, Nickname eingeben:\r\n");
@@ -167,12 +192,13 @@ void buy(BYTE n) {
                 * to NULL if no such credit could be found */
                credit = find_credit(nickname);
                if (credit != NULL) {
-                       if ((signed int)credit->credit < ((signed int)status.status[n].price * einheiten)) {
+                       if ((signed int)credit->credit < ((signed int)price * einheiten)) {
                                cprintf("Sorry, %s hat nicht genug Geld :-(\r\n", nickname);
-                               return;
+                               get_input();
+                               return 0;
                        }
-                       /* Geld abziehen */
-                       credit->credit -= (status.status[n].price * einheiten);
+                       /* substract money */
+                       credit->credit -= (price * einheiten);
                        cprintf("\r\nVerbleibendes Guthaben fuer %s: %d Cents. Druecke RETURN...\r\n",
                                nickname, credit->credit);
                        toggle_videomode();
@@ -183,18 +209,63 @@ void buy(BYTE n) {
                } else {
                        cprintf("\r\nNickname nicht gefunden in der Guthabenverwaltung! Abbruch, druecke RETURN...\r\n");
                        get_input();
-                       return;
+                       return 0;
                }
        } else {
                /* Ensure that nickname is NULL if it's empty because it's used in print_log */
                *nickname = '\0';
        }
        
-       status.status[n].times_sold += einheiten;
-       money += status.status[n].price * einheiten;
+       money += price * einheiten;
        items_sold += einheiten;
        if (printing == 1)
-               print_log(n, einheiten, nickname);
+               print_log(name, price, einheiten, nickname);
+
+       return einheiten;
+}
+
+void buy_stock(BYTE n) {
+       if (n >= status.num_items || status.status[n].item_name == NULL) {
+               cprintf("FEHLER: Diese Einheit existiert nicht.\r\n");
+               get_input();
+               return;
+       }
+
+       status.status[n].times_sold += buy(status.status[n].item_name, status.status[n].price);
+}
+
+void buy_custom() {
+       BYTE c = 0, i = 0;
+       int negative = 1;
+       char entered[5] = {'1', 0, 0, 0, 0};
+       char *input, name[10] = {0, 0, 0, 0, 0, 0, 0, 0, 0, 0};
+       int price;
+
+       cprintf("\r\nWas soll gekauft werden?\r\n");
+       input = get_input();
+       strncpy(name, input, 10);
+       if (*name == '\0')
+               return;
+
+       cprintf("\r\nWie teuer ist \"%s\" (in cents)?\r\n", name);
+       while (1) {
+               c = getchar();
+               if (c == 13)
+                       break;
+               else if (c == 27) {
+                       cprintf("Kauf abgebrochen, druecke RETURN...\r\n");
+                       get_input();
+                       return;
+               } else if (c == '-' && i == 0)
+                       negative = -1;
+               else if (c > 47 && c < 58)
+                       entered[i++] = c;
+       }
+       price = atoi(entered) * negative;
+
+       cprintf("\r\n");
+
+       buy(name, price);
 }
 
 void set_time_interactive() {
@@ -223,47 +294,47 @@ int main() {
 
        if (VIDEOMODE == 40)
                toggle_videomode();
-       /* Zeit erstmalig setzen */
+       /* Set time initially, c128 doesn't know it */
        set_time_interactive();
 
        POKE(216, 255);
 
-       /* Konfigurationsdatei laden */
+       /* Load configuration */
        load_config();
        cprintf("got %d logfiles\r\n", log_num);
 
-       /* Einträge (=Getränke) und Zustand laden */
+       /* Load items (= drinks) */
        load_items();
-       /* Guthaben laden */
+       /* Load credits */
        load_credits();
        while (1) {
-               /* Bildschirm anzeigen */
                print_screen();
-               /* Tastatureingaben abfragen */
                c = get_input();
-               /* und eventuell weitere Dialoge anzeigen */
+               /* ...display dialogs eventually */
                if (*c > 47 && *c < 58) {
-                       buy((*c) - 48);
+                       buy_stock((*c) - 48);
+                       toggle_videomode();
+                       clrscr();
+                       toggle_videomode();
+               } else if (*c == 'f') {
+                       buy_custom();
                        toggle_videomode();
                        clrscr();
                        toggle_videomode();
                } else if (*c == 's') {
-                       /* Zustandsdatei schreiben */
                        save_items();
                        save_credits();
                        cprintf("Statefile/Creditfile gesichert, druecke RETURN...\r\n");
                        get_input();
                } else if (*c == 'd') {
-                       /* Drucken an- oder ausschalten */
+                       /* enable/disable printing */
                        printing = (printing == 1 ? 0 : 1);
                        cprintf("Drucken ist nun %s, druecke RETURN...\r\n", 
                                (printing == 1 ? "eingeschaltet" : "ausgeschaltet"));
                        get_input();
                } else if (*c == 'g') {
-                       /* Guthabenverwalter aufrufen */
                        credit_manager();
                } else if (*c == 'z') {
-                       /* Zeit setzen */
                        set_time_interactive();
                } else if (*c == 'q')
                        break;