1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
|
/* $OpenBSD: magmareg.h,v 1.8 2006/03/04 13:00:55 miod Exp $ */
/* magmareg.h
*
* Copyright (c) 1998 Iain Hibbert
* All rights reserved.
*
* 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 and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
* 3. All advertising materials mentioning features or use of this software
* must display the following acknowledgement:
* This product includes software developed by Iain Hibbert
* 4. The name of the author may not be used to endorse or promote products
* derived from this software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``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 AUTHOR 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.
*
*/
#ifdef MAGMA_DEBUG
#define dprintf(x) printf x
#else
#define dprintf(x)
#endif
/* The mapping of minor device number -> card and port is done as
* follows by default:
*
* +---+---+---+---+---+---+---+---+
* | 7 | 6 | 5 | 4 | 3 | 2 | 1 | 0 |
* +---+---+---+---+---+---+---+---+
* | | | | | | | |
* | | | | +---+---+---+---> port number
* | | | |
* | | | +-------------------> dialout (on tty ports)
* | | |
* | | +-----------------------> unused
* | |
* +---+---------------------------> card number
*
*/
#define MAGMA_MAX_CARDS 4
#define MAGMA_MAX_TTY 16
#define MAGMA_MAX_BPP 2
#define MAGMA_MAX_CD1400 4
#define MAGMA_MAX_CD1190 2
#define MAGMA_CARD(x) ((minor(x) >> 6) & 0x03)
#define MAGMA_PORT(x) (minor(x) & 0x0f)
#define MTTY_DIALOUT(x) (minor(x) & 0x10)
/*
* Supported Card Types
*/
struct magma_board_info {
const char *mb_sbusname; /* sbus name */
const char *mb_name; /* cardname to match against */
const char *mb_realname; /* english card name */
int mb_nser; /* number of serial ports */
int mb_npar; /* number of parallel ports */
int mb_ncd1400; /* number of CD1400 chips */
int mb_svcackr; /* svcackr offset */
int mb_svcackt; /* svcackt offset */
int mb_svcackm; /* svcackm offset */
int mb_cd1400[MAGMA_MAX_CD1400];/* cd1400 chip register offsets */
int mb_ncd1190; /* number of CD1190 chips */
int mb_cd1190[MAGMA_MAX_CD1190];/* cd1190 chip register offsets */
};
/*
* cd1400 chip data
*/
struct cd1400 {
bus_space_handle_t cd_regh; /* chip register handle */
bus_space_tag_t cd_regt; /* chip register tag */
int cd_chiprev; /* chip revision */
int cd_clock; /* clock speed in MHz */
int cd_parmode; /* parallel mode operation */
};
/*
* cd1190 chip data
*/
struct cd1190 {
bus_space_handle_t cd_regh; /* chip register handle */
bus_space_tag_t cd_regt; /* chip register tag */
int cd_chiprev; /* chip revision */
};
/* software state for each card */
struct magma_softc {
struct device ms_dev; /* required. must be first in softc */
bus_space_tag_t sc_bustag; /* our bus tag */
bus_space_handle_t sc_iohandle; /* whole card registers */
void *sc_ih; /* interrupt vector */
void *sc_sih; /* softintr vector */
/* cd1400 chip info */
int ms_ncd1400;
struct cd1400 ms_cd1400[MAGMA_MAX_CD1400];
bus_space_handle_t sc_svcackrh; /* CD1400 service acknowledge receive */
bus_space_handle_t sc_svcackth; /* CD1400 service acknowledge transmit */
bus_space_handle_t sc_svcackmh; /* CD1400 service acknowledge modem */
/* cd1190 chip info */
int ms_ncd1190;
struct cd1190 ms_cd1190[MAGMA_MAX_CD1190];
const struct magma_board_info *ms_board; /* what am I? */
struct mtty_softc *ms_mtty;
struct mbpp_softc *ms_mbpp;
struct intrhand ms_hardint; /* hard interrupt handler */
struct intrhand ms_softint; /* soft interrupt handler */
};
#define MTTY_RBUF_SIZE (2 * 512)
#define MTTY_RX_FIFO_THRESHOLD 6
#define MTTY_RX_DTR_THRESHOLD 9
struct mtty_port {
struct cd1400 *mp_cd1400; /* ptr to chip */
int mp_channel; /* and channel */
struct tty *mp_tty;
int mp_openflags; /* default tty flags */
int mp_flags; /* port flags */
int mp_carrier; /* state of carrier */
u_char *mp_rbuf; /* ring buffer start */
u_char *mp_rend; /* ring buffer end */
u_char *mp_rget; /* ring buffer read pointer */
u_char *mp_rput; /* ring buffer write pointer */
u_char *mp_txp; /* transmit character pointer */
int mp_txc; /* transmit character counter */
};
#define MTTYF_CARRIER_CHANGED (1<<0)
#define MTTYF_SET_BREAK (1<<1)
#define MTTYF_CLR_BREAK (1<<2)
#define MTTYF_DONE (1<<3)
#define MTTYF_STOP (1<<4)
#define MTTYF_RING_OVERFLOW (1<<5)
struct mtty_softc {
struct device ms_dev; /* device info */
int ms_nports; /* tty ports */
struct mtty_port ms_port[MAGMA_MAX_TTY];
};
#define MBPP_RX_FIFO_THRESHOLD 25
struct mbpp_port {
struct cd1400 *mp_cd1400; /* for LC2+1Sp card */
struct cd1190 *mp_cd1190; /* all the others */
int mp_flags;
struct bpp_param mp_param;
#define mp_burst mp_param.bp_burst
#define mp_timeout mp_param.bp_timeout
#define mp_delay mp_param.bp_delay
u_char *mp_ptr; /* pointer to io data */
int mp_cnt; /* count of io chars */
struct timeout mp_timeout_tmo; /* for mbpp_timeout() */
struct timeout mp_start_tmo; /* for mbpp_start() */
};
#define MBPPF_OPEN (1<<0)
#define MBPPF_TIMEOUT (1<<1)
#define MBPPF_UIO (1<<2)
#define MBPPF_DELAY (1<<3)
#define MBPPF_WAKEUP (1<<4)
struct mbpp_softc {
struct device ms_dev; /* device info */
int ms_nports; /* parallel ports */
struct mbpp_port ms_port[MAGMA_MAX_BPP];
};
/* internal function prototypes */
int cd1400_compute_baud(speed_t, int, int *, int *);
__inline void cd1400_write_ccr(struct cd1400 *, u_char);
__inline u_char cd1400_read_reg(struct cd1400 *, int);
__inline void cd1400_write_reg(struct cd1400 *, int, u_char);
void cd1400_enable_transmitter(struct cd1400 *, int);
int magma_match(struct device *, void *, void *);
void magma_attach(struct device *, struct device *, void *);
int magma_hard(void *);
void magma_soft(void *);
int mtty_match(struct device *, void *, void *);
void mtty_attach(struct device *, struct device *, void *);
int mtty_modem_control(struct mtty_port *, int, int);
int mtty_param(struct tty *, struct termios *);
void mtty_start(struct tty *);
int mbpp_match(struct device *, void *, void *);
void mbpp_attach(struct device *, struct device *, void *);
int mbpp_rw(dev_t, struct uio *);
void mbpp_timeout(void *);
void mbpp_start(void *);
int mbpp_send(struct mbpp_port *, caddr_t, int);
int mbpp_recv(struct mbpp_port *, caddr_t, int);
int mbpp_hztoms(int);
int mbpp_mstohz(int);
#define CD1400_REGMAPSIZE 0x80
#define CD1190_REGMAPSIZE 0x20
|