Text in DOS mit einem Mal auf dem Bildschirm ausgeben



  • Laut PC-Intern bekommt man den aktuellen Videomodus mit

    _asm { 
            mov ah, 0fh   ;Videomode auslesen
            int 10h 
           } 
    
    Ergebnis:
      al: Videomode
      ah: Anzahl der Zeichen pro Zeile
      bh: aktuelle Bildschirmseite
    

    Harper schrieb:

    Textmodus mit 80 x 25 Zeichen (8 x 16 Pixel pro Zeichen) [oder] 80 x 50 Zeichen
    (8 x 8 Pixel pro Zeichen) wird vom Programm als Mode 3 interpretiert.

    Mode 3 sagt erstmal nur das es ein Textmode ist.
    Was auf den Sceen passt, ist abhängig vom eingestellten
    ROM-Zeichensatz.

    _asm { 
            mov ax, 1111h   ;8x14-ROM-Zeichensatz auswählen
            mov bl, 0h 
            int 10h 
           }
    

    oder

    _asm { 
            mov ax, 1112h   ;8x8-ROM-Zeichensatz auswählen
            mov bl, 0h 
            int 10h 
           }
    


  • Danke. Das mit dem Videomodus hat jetzt auch geklappt.

    Ich hab nun versucht, die Codepage einzustellen. Auf http://poli.cs.vsb.cz/misc/rbint/text/2112.html steht es folgendermaßen:

    INT 21 - DOS 3.3+ - GET GLOBAL CODE PAGE TABLE
    	AX = 6601h
    Return: CF set on error
    	    AX = error code (see #01680 at AH=59h/BX=0000h)
    	CF clear if successful
    	    BX = active code page (see #01757)
    	    DX = system code page (see #01757)
    SeeAlso: AX=6602h
    
    ----------
    
    INT 21 - DOS 3.3+ - SET GLOBAL CODE PAGE TABLE
    	AX = 6602h
    	BX = active code page (see #01757)
    	DX = system code page (active page at boot time)
    Return: CF set on error
    	    AX = error code (see #01680 at AH=59h/BX=0000h)
    	CF clear if successful
    	    AX = EB41h (Novell NWDOS v7.0 when NLSFUNC not installed and
    		  request was for previously-active code page)
    SeeAlso: AX=6601h,INT 2F/AX=14FFh
    

    Mein Code sieht so aus:

    #include <stdio.h>
    #include <dos.h>
    
    void get()
    {
    	union REGS in, out;
    
    	in.x.ax = 0x6601;
    	int86(0x21, &in, &out);
    
    	printf("Active code page: %i\n", out.x.bx);
    	printf("System code page: %i\n", out.x.dx);
    }
    
    void set()
    {
    	union REGS in, out;
    
    	in.x.ax = 0x6602;
    	in.x.bx = 850;
    	in.x.dx = 850;
    	int86(0x21, &in, &out);
    
    	printf("Code pages were set.\n");
    }
    
    int main()
    {
    	int i;
    	for (i = 128; i < 256; i++)
    		printf("%c", i);
    
    	printf("\n\n");
    
    	get();
    	set();
    	get();
    
    	return 0;
    }
    

    Das Ermitteln der Codepages geht. System-Codepage ist 437, active ist 850. Doch wenn ich dann beide auf 850 setze und sie mir danach noch mal ermitteln lasse, stehen sie immer noch auf 437 und 850. Das Setzen hat also nicht geklappt. Was mache ich falsch?



  • Wie dort steht, ist die "system code page" die code page, die beim Booten aktiv ist. Vermutlich die im BIOS eingestellte code page. Die kannst du iaR. nicht ueber diese Funktion umstellen. ZB. die Doku von DR-DOS fuehrt diese Moeglichkeit daher konsequenterweise gar nicht erst auf.



  • Ja, das macht Sinn. Aber auch wenn ich Zeile 21 auskommentiere (und in Zeile 20 dann 437 hinschreibe, weil 850 ja schon gesetzt ist), bleibt die Active Code Page so wie sie war.



  • Sehe da sonst gerade keinen groben Fehler...
    Checke die Fehlercodes (CF und AX) und stelle sicher, dass die ganzen Geschichten mit country.sys und das mode select Zeugs in deinem DOS korrekt eingerichtet sind, bzw. die Funktion ueberhaupt unterstuetzt wird.



  • Was soll eigentlich CF sein? Ich lese das ständig, aber REGS besitzt keine Variable CF. Was muss ich da genau abprüfen?

    Nobuo T schrieb:

    und stelle sicher, dass die ganzen Geschichten mit country.sys und das mode select Zeugs in deinem DOS korrekt eingerichtet sind, bzw. die Funktion ueberhaupt unterstuetzt wird.

    Na ja, ich arbeite nicht unbedingt unter einem richtigen DOS, sondern benutze meistens die Konsole in Windows XP, die das Programm ja durchaus abspielen kann. Ich denke nicht, dass ich da im Vorfeld irgendwie großartig was einstellen kann. Ansonsten habe ich auch noch DOSBox.
    Gibt es für das, was du hier erwähnst, eine Möglichkeit, es im Programm selbst zu realisieren? Ich möchte ungern ein Programm schreiben, bei dem ich dem Benutzer erst erklären muss, was er vorher einstellen soll. Das soll das Programm am Anfang selbst machen und am besten am Ende wieder zurückstellen.



  • Harper schrieb:

    Was soll eigentlich CF sein? Ich lese das ständig, aber REGS besitzt keine Variable CF. Was muss ich da genau abprüfen?

    CF ist das Carry-Flag aus dem Status-Register.



  • Es sollte in den REGS die Variable "cflag" geben, wenn ich mich nicht irre. Sonst sollte REGS.x.flags & 1 funktionieren.
    Habe zu dieser Gelegenheit mal seit Langem wieder schnell ein kleines ASM-Programm zum Testen der Code Pages hingehackt (http://btm.homeip.net/cppde/codept.com wen's interessiert 300+x Byte, Haelfte Text -> fuer Quelltext einen Disassembler bemuehen). Ergebnis: Die DOS-VM von Windows XP kennt wohl Funktion 6602 nicht (gibt Fehler 01).
    Falls du also mit Code Pages spielen willst, versuch's mit der DOS Box oder einem echten DOS.



  • An Windows XP scheint's nicht zu liegen. Ich hab noch mal folgenden Code geschrieben:

    #include <stdio.h>
    #include <dos.h>
    
    void int86func(int intno, union REGS *in, union REGS *out)
    {
    	int retval = int86(intno, in, out);
    
    	printf("int86 returned:   %i\n", retval);
    	printf("REGS.x.cflags:    %i\n", out->x.cflag);
    	printf("REGS.x.flags & 1: %i\n", out->x.flags & 1);
    }
    
    void get()
    {
        union REGS in, out;
    
        printf("GET\n");
    
        in.x.ax = 0x6601;
        int86func(0x21, &in, &out);
    
        printf("Active code page: %i\n", out.x.bx);
        printf("System code page: %i\n\n", out.x.dx);
    }
    
    void set(unsigned int page)
    {
        union REGS in, out;
    
        printf("SET active code page to %i\n", page);
    
        in.x.ax = 0x6602;
        in.x.bx = page;
        int86func(0x21, &in, &out);
    
        printf("\n");
    
        get();
    }
    
    int main()
    {
        int i;
    
        for (i = 128; i < 256; i++)
    	printf("%c", i);
    
        printf("\n\n");
    
        get();
        set(850);
        set(437);
    
        return 0;
    }
    

    Und egal ob ich das ganze in Windows XP ausführe oder in DOSBox, Windows 98, dem DOS-Modus von Windows 98 oder FreeDOS (alles außer XP läuft bei mir in Virtual PC), das get liefert bei cflag zwar 0, aber das set liefert immer den Fehlercode 1. Da muss also noch irgend etwas falsch sein bzw. fehlen. Auch mit deinem Programm hat es in noch keiner Konfiguration geklappt, die Codepage tatsächlich zu setzen.



  • ... vielleicht hilft dir das hier auch noch weiter... http://www.drdos.net/documentation/sysprog/chap4.htm



  • Meinst du irgendetwas spezielles in dem Dokument?



  • Den Abschnitt ueber Funktion 66?



  • Das einzig Neue bzw. Hilfreiche war dort der Satz:

    NLSFUNC must be loaded to change the global code page.

    Wenn ich das mache, geht's nämlich. Aber insgesamt ist das doch auch blöd: Soll ich jetzt in meinem Programm etwa system("nlsfunc"); schreiben? Zumal es diese Datei zum Beispiel in DOSBox gar nicht gibt. Die hätten mal lieber hinschreiben sollen, was man quellcodetechnisch machen muss, um das Ändern der Codepage zu ermöglichen.



  • Quelltext ist wohl nicht das grosse Problem:

    http://www.ibiblio.org/pub/micro/pc-stuff/freedos/files/dos/nlsfunc/nlsfn04s.zip

    http://www.ibiblio.org/pub/micro/pc-stuff/freedos/files/dos/nlsfunc/

    , sondern eher die Aussage:

    Verwenden Sie den Befehl NLSFUNC nicht, wenn Sie Windows gestartet haben. Er könnte Ihren Computer daran hindern, richtig zu funktionieren.
    

    http://www.i8086.de/dos-befehle/nlsfunc.html

    .



  • Tja, da meine Programme möglichst auch unter Windows korrekt funktionieren sollen, muss ich erneut fragen: Was kann man da machen? Ist es denn wirklich nicht möglich, mal eben schnell die Codepage umzustellen und am Ende wieder zurück zu stellen?

    Alles andere lässt sich doch auch einstellen: Bildschirmmodus, Auflösung und sogar die Frage, ob Zeichen blinken können sollen oder ob man lieber 16 statt 8 Hintergrundfarben haben kann. Alles kann man mit einem Mal ändern. Warum ist das bei der Codepage so schwierig?

    Ist das letztendlich nicht einfach nur eine Bitmap, die dort ausgetauscht wird? Und gibt es denn wirklich keine Programme, wo das mal tatsächlich gemacht wird, so dass man dort nachgucken kann? Ist es wirklich so, dass man nur Stückchen und Brocken in irgendwelchen Dokus findet, wie man diesen Befehl theoretisch aufrufen könnte, wenn man das und das und das eingestellt hat, statt mal ein Beispiel zu zeigen, was ein Programm machen muss, um die Codepage zu setzen und am Ende wieder zurück zu setzen?



  • Keiner eine Idee?


Anmelden zum Antworten