summaryrefslogtreecommitdiff
path: root/drivers/s390/char/con3270.c
diff options
context:
space:
mode:
Diffstat (limited to 'drivers/s390/char/con3270.c')
-rw-r--r--drivers/s390/char/con3270.c35
1 files changed, 21 insertions, 14 deletions
diff --git a/drivers/s390/char/con3270.c b/drivers/s390/char/con3270.c
index 1c1390249342..d3c9fb27a03f 100644
--- a/drivers/s390/char/con3270.c
+++ b/drivers/s390/char/con3270.c
@@ -36,6 +36,8 @@
#define TTY3270_OUTPUT_BUFFER_SIZE 1024
#define TTY3270_STRING_PAGES 5
+#define TTY3270_SCREEN_PAGES 8 /* has to be power-of-two */
+
static struct tty_driver *tty3270_driver;
static int tty3270_max_index;
static struct tty3270 *condev;
@@ -56,6 +58,7 @@ struct tty3270_cell {
struct tty3270_line {
struct tty3270_cell *cells;
int len;
+ int dirty;
};
static const unsigned char sfq_read_partition[] = {
@@ -92,6 +95,7 @@ struct tty3270 {
unsigned int cx, cy; /* Current output position. */
struct tty3270_attribute attributes;
struct tty3270_attribute saved_attributes;
+ int allocated_lines;
struct tty3270_line *screen;
/* Input stuff. */
@@ -805,22 +809,22 @@ static void tty3270_free_view(struct tty3270 *tp)
/*
* Allocate tty3270 screen.
*/
-static struct tty3270_line *tty3270_alloc_screen(unsigned int rows, unsigned int cols)
+static struct tty3270_line *tty3270_alloc_screen(struct tty3270 *tp, unsigned int rows,
+ unsigned int cols, int *allocated_out)
{
struct tty3270_line *screen;
- unsigned long size;
- int lines;
+ int allocated, lines;
- size = sizeof(struct tty3270_line) * (rows - 2);
- screen = kzalloc(size, GFP_KERNEL);
+ allocated = __roundup_pow_of_two(rows) * TTY3270_SCREEN_PAGES;
+ screen = kcalloc(allocated, sizeof(struct tty3270_line), GFP_KERNEL);
if (!screen)
goto out_err;
- for (lines = 0; lines < rows - 2; lines++) {
- size = sizeof(struct tty3270_cell) * cols;
- screen[lines].cells = kzalloc(size, GFP_KERNEL);
+ for (lines = 0; lines < allocated; lines++) {
+ screen[lines].cells = kcalloc(cols, sizeof(struct tty3270_cell), GFP_KERNEL);
if (!screen[lines].cells)
goto out_screen;
}
+ *allocated_out = allocated;
return screen;
out_screen:
while (lines--)
@@ -833,11 +837,11 @@ out_err:
/*
* Free tty3270 screen.
*/
-static void tty3270_free_screen(struct tty3270_line *screen, unsigned int rows)
+static void tty3270_free_screen(struct tty3270_line *screen, int old_lines)
{
int lines;
- for (lines = 0; lines < rows - 2; lines++)
+ for (lines = 0; lines < old_lines; lines++)
kfree(screen[lines].cells);
kfree(screen);
}
@@ -853,6 +857,7 @@ static void tty3270_resize(struct raw3270_view *view,
struct tty3270_line *screen, *oscreen;
struct tty_struct *tty;
struct winsize ws;
+ int new_allocated, old_allocated = tp->allocated_lines;
if (old_model == new_model &&
old_cols == new_cols &&
@@ -863,7 +868,7 @@ static void tty3270_resize(struct raw3270_view *view,
spin_unlock_irq(&tp->view.lock);
return;
}
- screen = tty3270_alloc_screen(new_rows, new_cols);
+ screen = tty3270_alloc_screen(tp, new_rows, new_cols, &new_allocated);
if (IS_ERR(screen))
return;
/* Switch to new output size */
@@ -871,6 +876,7 @@ static void tty3270_resize(struct raw3270_view *view,
tty3270_blank_screen(tp);
oscreen = tp->screen;
tp->screen = screen;
+ tp->allocated_lines = new_allocated;
tp->view.rows = new_rows;
tp->view.cols = new_cols;
tp->view.model = new_model;
@@ -883,7 +889,7 @@ static void tty3270_resize(struct raw3270_view *view,
tty3270_blank_line(tp);
tp->update_flags = TTY_UPDATE_ALL;
spin_unlock_irq(&tp->view.lock);
- tty3270_free_screen(oscreen, old_rows);
+ tty3270_free_screen(oscreen, old_allocated);
tty3270_set_timer(tp, 1);
/* Informat tty layer about new size */
tty = tty_port_tty_get(&tp->port);
@@ -920,7 +926,7 @@ static void tty3270_free(struct raw3270_view *view)
struct tty3270 *tp = container_of(view, struct tty3270, view);
del_timer_sync(&tp->timer);
- tty3270_free_screen(tp->screen, tp->view.rows);
+ tty3270_free_screen(tp->screen, tp->allocated_lines);
tty3270_free_view(tp);
}
@@ -969,7 +975,8 @@ tty3270_create_view(int index, struct tty3270 **newtp)
return rc;
}
- tp->screen = tty3270_alloc_screen(tp->view.rows, tp->view.cols);
+ tp->screen = tty3270_alloc_screen(tp, tp->view.rows, tp->view.cols,
+ &tp->allocated_lines);
if (IS_ERR(tp->screen)) {
rc = PTR_ERR(tp->screen);
raw3270_put_view(&tp->view);