Newer
Older
UbixOS / sys / kernel / tty.c
/*-
 * Copyright (c) 2002-2018 The UbixOS Project.
 * All rights reserved.
 *
 * This was developed by Christopher W. Olsen for the UbixOS Project.
 *
 * Redistribution and use in source and binary forms, with or without modification, are permitted
 * provided that the following conditions are met:
 *
 * 1) Redistributions of source code must retain the above copyright notice, this list of
 *    conditions, the following disclaimer and the list of authors.
 * 2) Redistributions in binary form must reproduce the above copyright notice, this list of
 *    conditions, the following disclaimer and the list of authors in the documentation and/or
 *    other materials provided with the distribution.
 * 3) Neither the name of the UbixOS Project nor the names of its contributors may be used to
 *    endorse or promote products derived from this software without specific prior written
 *    permission.
 *
 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED
 * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
 * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS
 * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
 * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA,
 * OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
 * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT
 * OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
 */

#include <ubixos/tty.h>
#include <ubixos/kpanic.h>
#include <ubixos/spinlock.h>
#include <lib/kprintf.h>
#include <lib/kmalloc.h>
#include <sys/io.h>
#include <string.h>

static tty_term *terms = 0x0;
tty_term *tty_foreground = 0x0;
static struct spinLock tty_spinLock = SPIN_LOCK_INITIALIZER;

int tty_init() {
  int i = 0x0;

  /* Allocate memory for terminals */
  terms = (tty_term *) kmalloc(sizeof(tty_term) * TTY_MAX_TERMS);
  if (terms == 0x0)
    kpanic("tty_init: Failed to allocate memory. File: %s, Line: %i\n", __FILE__, __LINE__);

  /* Set up all default terminal information */
  for (i = 0; i < TTY_MAX_TERMS; i++) {
    terms[i].tty_buffer = (char *) kmalloc(80 * 60 * 2);
    if (terms[i].tty_buffer == 0x0)
      kpanic("tty_init: Failed to allocate buffer memory. File: %s, Line: %i\n", __FILE__, __LINE__);

    terms[i].tty_pointer = terms[i].tty_buffer; /* Set up tty pointer to internal buffer */
    terms[i].tty_x = 0x0; /* Set up default X position             */
    terms[i].tty_y = 0x0; /* Set up default Y position             */
    terms[i].tty_colour = 0x0A + i; /* Set up default tty text colour        */
  }

  /* Read tty0 current position (to migrate from kprintf). */
  outportByte(0x3D4, 0x0e);
  terms[0].tty_y = inportByte(0x3D5);
  outportByte(0x3D4, 0x0f);
  terms[0].tty_x = inportByte(0x3D5);

  /* Set up pointer for the foreground tty */
  tty_foreground = &terms[0];

  /* Set up the foreground ttys information */
  tty_foreground->tty_pointer = (char *) 0xB8000;

  /* Return to let kernel know initialization is complete */
  kprintf("tty0 - Initialized\n");

  return (0x0);
}

/*
 This will change the specified tty. It ultimately copies the screen
 to the foreground buffer copies the new ttys buffer to the screen and
 adjusts a couple pointers and we are good to go.
 */
int tty_change(uInt16 tty) {

  if (tty > TTY_MAX_TERMS)
    kpanic("Error: Changing to an invalid tty. File: %s, Line: %i\n", __FILE__, __LINE__);

  /* Copy display buffer to tty buffer */
  memcpy(tty_foreground->tty_buffer, (char *) 0xB8000, (80 * 60 * 2));

  /* Copy new tty buffer to display buffer */
  memcpy((char *) 0xB8000, terms[tty].tty_buffer, (80 * 60 * 2));

  /*
   Set the tty_pointer to the internal buffer so I can continue 
   writing to what it believes is the screen
   */
  tty_foreground->tty_pointer = tty_foreground->tty_buffer;

  terms[tty].tty_pointer = (char *) 0xB8000;

  /* set new foreground tty */
  tty_foreground = &terms[tty];

  /* Adjust cursor when we change consoles */
  outportByte(0x3D4, 0x0F);
  outportByte(0x3D5, tty_foreground->tty_x);
  outportByte(0x3D4, 0x0E);
  outportByte(0x3D5, tty_foreground->tty_y);

  return (0x0);
}

int tty_print(char *string, tty_term *term) {
  unsigned int bufferOffset = 0x0, character = 0x0, i = 0x0;
  spinLock(&tty_spinLock);

  /* We Need To Get The Y Position */
  bufferOffset = term->tty_y;
  bufferOffset <<= 8;

  /* Then We Need To Add The X Position */
  bufferOffset += term->tty_x;
  bufferOffset <<= 1;

  while ((character = *string++)) {
    switch (character) {
      case '\n':
        bufferOffset = (bufferOffset / 160) * 160 + 160;
      break;
      default:
        term->tty_pointer[bufferOffset++] = character;
        term->tty_pointer[bufferOffset++] = term->tty_colour;
      break;
    } /* switch */

    /* Check To See If We Are Out Of Bounds */
    if (bufferOffset >= 160 * 25) {
      for (i = 0; i < 160 * 24; i++) {
        term->tty_pointer[i] = term->tty_pointer[i + 160];
      }
      for (i = 0; i < 80; i++) {
        term->tty_pointer[(160 * 24) + (i * 2)] = 0x20;
        term->tty_pointer[(160 * 24) + (i * 2) + 1] = term->tty_colour;
      }
      bufferOffset -= 160;
    }
  }

  bufferOffset >>= 1; /* Set the new cursor position  */
  term->tty_x = (bufferOffset & 0xFF);
  term->tty_y = (bufferOffset >> 0x8);

  if (term == tty_foreground) {
    outportByte(0x3D4, 0x0f);
    outportByte(0x3D5, term->tty_x);
    outportByte(0x3D4, 0x0e);
    outportByte(0x3D5, term->tty_y);
  }

  spinUnlock(&tty_spinLock);

  return (0x0);
}

tty_term *tty_find(uInt16 tty) {
  return (&terms[tty]);
}

/***
 END
 ***/