Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 1 | /* |
Martin Schwidefsky | 4b214a0 | 2009-06-16 10:30:47 +0200 | [diff] [blame] | 2 | * IBM/3270 Driver |
Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 3 | * |
Martin Schwidefsky | 4b214a0 | 2009-06-16 10:30:47 +0200 | [diff] [blame] | 4 | * Author(s): |
| 5 | * Original 3270 Code for 2.4 written by Richard Hitt (UTS Global) |
| 6 | * Rewritten for 2.5 by Martin Schwidefsky <schwidefsky@de.ibm.com> |
| 7 | * Copyright IBM Corp. 2003, 2009 |
Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 8 | */ |
| 9 | |
| 10 | #include <asm/idals.h> |
| 11 | #include <asm/ioctl.h> |
| 12 | |
| 13 | /* ioctls for fullscreen 3270 */ |
| 14 | #define TUBICMD _IO('3', 3) /* set ccw command for fs reads. */ |
| 15 | #define TUBOCMD _IO('3', 4) /* set ccw command for fs writes. */ |
| 16 | #define TUBGETI _IO('3', 7) /* get ccw command for fs reads. */ |
| 17 | #define TUBGETO _IO('3', 8) /* get ccw command for fs writes. */ |
| 18 | #define TUBSETMOD _IO('3',12) /* FIXME: what does it do ?*/ |
| 19 | #define TUBGETMOD _IO('3',13) /* FIXME: what does it do ?*/ |
| 20 | |
| 21 | /* Local Channel Commands */ |
| 22 | #define TC_WRITE 0x01 /* Write */ |
Richard Hitt | ed3cb6f | 2005-10-30 15:00:10 -0800 | [diff] [blame] | 23 | #define TC_RDBUF 0x02 /* Read Buffer */ |
Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 24 | #define TC_EWRITE 0x05 /* Erase write */ |
| 25 | #define TC_READMOD 0x06 /* Read modified */ |
| 26 | #define TC_EWRITEA 0x0d /* Erase write alternate */ |
| 27 | #define TC_WRITESF 0x11 /* Write structured field */ |
| 28 | |
| 29 | /* Buffer Control Orders */ |
| 30 | #define TO_SF 0x1d /* Start field */ |
| 31 | #define TO_SBA 0x11 /* Set buffer address */ |
| 32 | #define TO_IC 0x13 /* Insert cursor */ |
| 33 | #define TO_PT 0x05 /* Program tab */ |
| 34 | #define TO_RA 0x3c /* Repeat to address */ |
| 35 | #define TO_SFE 0x29 /* Start field extended */ |
| 36 | #define TO_EUA 0x12 /* Erase unprotected to address */ |
| 37 | #define TO_MF 0x2c /* Modify field */ |
| 38 | #define TO_SA 0x28 /* Set attribute */ |
| 39 | |
| 40 | /* Field Attribute Bytes */ |
| 41 | #define TF_INPUT 0x40 /* Visible input */ |
| 42 | #define TF_INPUTN 0x4c /* Invisible input */ |
| 43 | #define TF_INMDT 0xc1 /* Visible, Set-MDT */ |
| 44 | #define TF_LOG 0x60 |
| 45 | |
| 46 | /* Character Attribute Bytes */ |
| 47 | #define TAT_RESET 0x00 |
| 48 | #define TAT_FIELD 0xc0 |
| 49 | #define TAT_EXTHI 0x41 |
| 50 | #define TAT_COLOR 0x42 |
| 51 | #define TAT_CHARS 0x43 |
| 52 | #define TAT_TRANS 0x46 |
| 53 | |
| 54 | /* Extended-Highlighting Bytes */ |
| 55 | #define TAX_RESET 0x00 |
| 56 | #define TAX_BLINK 0xf1 |
| 57 | #define TAX_REVER 0xf2 |
| 58 | #define TAX_UNDER 0xf4 |
| 59 | |
| 60 | /* Reset value */ |
| 61 | #define TAR_RESET 0x00 |
| 62 | |
| 63 | /* Color values */ |
| 64 | #define TAC_RESET 0x00 |
| 65 | #define TAC_BLUE 0xf1 |
| 66 | #define TAC_RED 0xf2 |
| 67 | #define TAC_PINK 0xf3 |
| 68 | #define TAC_GREEN 0xf4 |
| 69 | #define TAC_TURQ 0xf5 |
| 70 | #define TAC_YELLOW 0xf6 |
| 71 | #define TAC_WHITE 0xf7 |
| 72 | #define TAC_DEFAULT 0x00 |
| 73 | |
| 74 | /* Write Control Characters */ |
| 75 | #define TW_NONE 0x40 /* No particular action */ |
| 76 | #define TW_KR 0xc2 /* Keyboard restore */ |
| 77 | #define TW_PLUSALARM 0x04 /* Add this bit for alarm */ |
| 78 | |
Richard Hitt | ed3cb6f | 2005-10-30 15:00:10 -0800 | [diff] [blame] | 79 | #define RAW3270_FIRSTMINOR 1 /* First minor number */ |
| 80 | #define RAW3270_MAXDEVS 255 /* Max number of 3270 devices */ |
Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 81 | |
| 82 | /* For TUBGETMOD and TUBSETMOD. Should include. */ |
| 83 | struct raw3270_iocb { |
| 84 | short model; |
| 85 | short line_cnt; |
| 86 | short col_cnt; |
| 87 | short pf_cnt; |
| 88 | short re_cnt; |
| 89 | short map; |
| 90 | }; |
| 91 | |
| 92 | struct raw3270; |
| 93 | struct raw3270_view; |
Martin Schwidefsky | c95571e | 2013-01-08 15:31:11 +0100 | [diff] [blame] | 94 | extern struct class *class3270; |
Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 95 | |
| 96 | /* 3270 CCW request */ |
| 97 | struct raw3270_request { |
| 98 | struct list_head list; /* list head for request queueing. */ |
| 99 | struct raw3270_view *view; /* view of this request */ |
| 100 | struct ccw1 ccw; /* single ccw. */ |
| 101 | void *buffer; /* output buffer. */ |
| 102 | size_t size; /* size of output buffer. */ |
| 103 | int rescnt; /* residual count from devstat. */ |
| 104 | int rc; /* return code for this request. */ |
| 105 | |
| 106 | /* Callback for delivering final status. */ |
| 107 | void (*callback)(struct raw3270_request *, void *); |
| 108 | void *callback_data; |
| 109 | }; |
| 110 | |
| 111 | struct raw3270_request *raw3270_request_alloc(size_t size); |
| 112 | struct raw3270_request *raw3270_request_alloc_bootmem(size_t size); |
| 113 | void raw3270_request_free(struct raw3270_request *); |
| 114 | void raw3270_request_reset(struct raw3270_request *); |
| 115 | void raw3270_request_set_cmd(struct raw3270_request *, u8 cmd); |
| 116 | int raw3270_request_add_data(struct raw3270_request *, void *, size_t); |
| 117 | void raw3270_request_set_data(struct raw3270_request *, void *, size_t); |
| 118 | void raw3270_request_set_idal(struct raw3270_request *, struct idal_buffer *); |
| 119 | |
| 120 | static inline int |
| 121 | raw3270_request_final(struct raw3270_request *rq) |
| 122 | { |
| 123 | return list_empty(&rq->list); |
| 124 | } |
| 125 | |
| 126 | void raw3270_buffer_address(struct raw3270 *, char *, unsigned short); |
| 127 | |
Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 128 | /* |
| 129 | * Functions of a 3270 view. |
| 130 | */ |
| 131 | struct raw3270_fn { |
| 132 | int (*activate)(struct raw3270_view *); |
| 133 | void (*deactivate)(struct raw3270_view *); |
Martin Schwidefsky | 8340ab6 | 2016-05-02 14:53:29 +0200 | [diff] [blame] | 134 | void (*intv)(struct raw3270_view *, |
Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 135 | struct raw3270_request *, struct irb *); |
| 136 | void (*release)(struct raw3270_view *); |
| 137 | void (*free)(struct raw3270_view *); |
Martin Schwidefsky | 4d334fd | 2013-01-04 14:55:13 +0100 | [diff] [blame] | 138 | void (*resize)(struct raw3270_view *, int, int, int); |
Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 139 | }; |
| 140 | |
| 141 | /* |
| 142 | * View structure chaining. The raw3270_view structure is meant to |
| 143 | * be embedded at the start of the real view data structure, e.g.: |
| 144 | * struct example { |
| 145 | * struct raw3270_view view; |
| 146 | * ... |
| 147 | * }; |
| 148 | */ |
| 149 | struct raw3270_view { |
| 150 | struct list_head list; |
| 151 | spinlock_t lock; |
Martin Schwidefsky | 00853c9 | 2019-04-03 09:13:34 +0200 | [diff] [blame] | 152 | #define RAW3270_VIEW_LOCK_IRQ 0 |
| 153 | #define RAW3270_VIEW_LOCK_BH 1 |
Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 154 | atomic_t ref_count; |
| 155 | struct raw3270 *dev; |
| 156 | struct raw3270_fn *fn; |
| 157 | unsigned int model; |
| 158 | unsigned int rows, cols; /* # of rows & colums of the view */ |
| 159 | unsigned char *ascebc; /* ascii -> ebcdic table */ |
| 160 | }; |
| 161 | |
Martin Schwidefsky | 00853c9 | 2019-04-03 09:13:34 +0200 | [diff] [blame] | 162 | int raw3270_add_view(struct raw3270_view *, struct raw3270_fn *, int, int); |
Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 163 | int raw3270_activate_view(struct raw3270_view *); |
| 164 | void raw3270_del_view(struct raw3270_view *); |
| 165 | void raw3270_deactivate_view(struct raw3270_view *); |
| 166 | struct raw3270_view *raw3270_find_view(struct raw3270_fn *, int); |
| 167 | int raw3270_start(struct raw3270_view *, struct raw3270_request *); |
Richard Hitt | ed3cb6f | 2005-10-30 15:00:10 -0800 | [diff] [blame] | 168 | int raw3270_start_locked(struct raw3270_view *, struct raw3270_request *); |
Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 169 | int raw3270_start_irq(struct raw3270_view *, struct raw3270_request *); |
Richard Hitt | ed3cb6f | 2005-10-30 15:00:10 -0800 | [diff] [blame] | 170 | int raw3270_reset(struct raw3270_view *); |
| 171 | struct raw3270_view *raw3270_view(struct raw3270_view *); |
Martin Schwidefsky | 233faec | 2014-03-21 09:25:24 +0100 | [diff] [blame] | 172 | int raw3270_view_active(struct raw3270_view *); |
Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 173 | |
| 174 | /* Reference count inliner for view structures. */ |
| 175 | static inline void |
| 176 | raw3270_get_view(struct raw3270_view *view) |
| 177 | { |
| 178 | atomic_inc(&view->ref_count); |
| 179 | } |
| 180 | |
| 181 | extern wait_queue_head_t raw3270_wait_queue; |
| 182 | |
| 183 | static inline void |
| 184 | raw3270_put_view(struct raw3270_view *view) |
| 185 | { |
| 186 | if (atomic_dec_return(&view->ref_count) == 0) |
| 187 | wake_up(&raw3270_wait_queue); |
| 188 | } |
| 189 | |
Sebastian Ott | 2253e8d | 2014-01-27 13:26:10 +0100 | [diff] [blame] | 190 | struct raw3270 *raw3270_setup_console(void); |
Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 191 | void raw3270_wait_cons_dev(struct raw3270 *); |
| 192 | |
| 193 | /* Notifier for device addition/removal */ |
Martin Schwidefsky | c95571e | 2013-01-08 15:31:11 +0100 | [diff] [blame] | 194 | struct raw3270_notifier { |
| 195 | struct list_head list; |
| 196 | void (*create)(int minor); |
| 197 | void (*destroy)(int minor); |
| 198 | }; |
| 199 | |
| 200 | int raw3270_register_notifier(struct raw3270_notifier *); |
| 201 | void raw3270_unregister_notifier(struct raw3270_notifier *); |
Martin Schwidefsky | 4b214a0 | 2009-06-16 10:30:47 +0200 | [diff] [blame] | 202 | void raw3270_pm_unfreeze(struct raw3270_view *); |
Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 203 | |
| 204 | /* |
| 205 | * Little memory allocator for string objects. |
| 206 | */ |
| 207 | struct string |
| 208 | { |
| 209 | struct list_head list; |
| 210 | struct list_head update; |
| 211 | unsigned long size; |
| 212 | unsigned long len; |
| 213 | char string[0]; |
| 214 | } __attribute__ ((aligned(8))); |
| 215 | |
| 216 | static inline struct string * |
| 217 | alloc_string(struct list_head *free_list, unsigned long len) |
| 218 | { |
| 219 | struct string *cs, *tmp; |
| 220 | unsigned long size; |
| 221 | |
| 222 | size = (len + 7L) & -8L; |
| 223 | list_for_each_entry(cs, free_list, list) { |
| 224 | if (cs->size < size) |
| 225 | continue; |
| 226 | if (cs->size > size + sizeof(struct string)) { |
| 227 | char *endaddr = (char *) (cs + 1) + cs->size; |
| 228 | tmp = (struct string *) (endaddr - size) - 1; |
| 229 | tmp->size = size; |
| 230 | cs->size -= size + sizeof(struct string); |
| 231 | cs = tmp; |
| 232 | } else |
| 233 | list_del(&cs->list); |
| 234 | cs->len = len; |
| 235 | INIT_LIST_HEAD(&cs->list); |
| 236 | INIT_LIST_HEAD(&cs->update); |
| 237 | return cs; |
| 238 | } |
Heiko Carstens | d2c993d | 2006-07-12 16:41:55 +0200 | [diff] [blame] | 239 | return NULL; |
Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 240 | } |
| 241 | |
| 242 | static inline unsigned long |
| 243 | free_string(struct list_head *free_list, struct string *cs) |
| 244 | { |
| 245 | struct string *tmp; |
| 246 | struct list_head *p, *left; |
| 247 | |
| 248 | /* Find out the left neighbour in free memory list. */ |
| 249 | left = free_list; |
| 250 | list_for_each(p, free_list) { |
| 251 | if (list_entry(p, struct string, list) > cs) |
| 252 | break; |
| 253 | left = p; |
| 254 | } |
| 255 | /* Try to merge with right neighbour = next element from left. */ |
| 256 | if (left->next != free_list) { |
| 257 | tmp = list_entry(left->next, struct string, list); |
| 258 | if ((char *) (cs + 1) + cs->size == (char *) tmp) { |
| 259 | list_del(&tmp->list); |
| 260 | cs->size += tmp->size + sizeof(struct string); |
| 261 | } |
| 262 | } |
| 263 | /* Try to merge with left neighbour. */ |
| 264 | if (left != free_list) { |
| 265 | tmp = list_entry(left, struct string, list); |
| 266 | if ((char *) (tmp + 1) + tmp->size == (char *) cs) { |
| 267 | tmp->size += cs->size + sizeof(struct string); |
| 268 | return tmp->size; |
| 269 | } |
| 270 | } |
| 271 | __list_add(&cs->list, left, left->next); |
| 272 | return cs->size; |
| 273 | } |
| 274 | |
| 275 | static inline void |
| 276 | add_string_memory(struct list_head *free_list, void *mem, unsigned long size) |
| 277 | { |
| 278 | struct string *cs; |
| 279 | |
| 280 | cs = (struct string *) mem; |
| 281 | cs->size = size - sizeof(struct string); |
| 282 | free_string(free_list, cs); |
| 283 | } |
| 284 | |