1 // SPDX-License-Identifier: GPL-2.0
3 * SCLP Event Type (ET) 7 - Diagnostic Test FTP Services, useable on LPAR
5 * Copyright IBM Corp. 2013
6 * Author(s): Ralf Hoppe (rhoppe@de.ibm.com)
10 #define KMSG_COMPONENT "hmcdrv"
11 #define pr_fmt(fmt) KMSG_COMPONENT ": " fmt
13 #include <linux/kernel.h>
15 #include <linux/slab.h>
17 #include <linux/wait.h>
18 #include <linux/string.h>
19 #include <linux/jiffies.h>
20 #include <asm/sysinfo.h>
21 #include <asm/ebcdic.h>
24 #include "sclp_diag.h"
27 static DECLARE_COMPLETION(sclp_ftp_rx_complete
);
28 static u8 sclp_ftp_ldflg
;
29 static u64 sclp_ftp_fsize
;
30 static u64 sclp_ftp_length
;
33 * sclp_ftp_txcb() - Diagnostic Test FTP services SCLP command callback
35 * @data: pointer to struct completion
37 static void sclp_ftp_txcb(struct sclp_req
*req
, void *data
)
39 struct completion
*completion
= data
;
42 pr_debug("SCLP (ET7) TX-IRQ, SCCB @ 0x%p: %*phN\n",
43 req
->sccb
, 24, req
->sccb
);
49 * sclp_ftp_rxcb() - Diagnostic Test FTP services receiver event callback
50 * @evbuf: pointer to Diagnostic Test (ET7) event buffer
52 static void sclp_ftp_rxcb(struct evbuf_header
*evbuf
)
54 struct sclp_diag_evbuf
*diag
= (struct sclp_diag_evbuf
*) evbuf
;
57 * Check for Diagnostic Test FTP Service
59 if (evbuf
->type
!= EVTYP_DIAG_TEST
||
60 diag
->route
!= SCLP_DIAG_FTP_ROUTE
||
61 diag
->mdd
.ftp
.pcx
!= SCLP_DIAG_FTP_XPCX
||
62 evbuf
->length
< SCLP_DIAG_FTP_EVBUF_LEN
)
66 pr_debug("SCLP (ET7) RX-IRQ, Event @ 0x%p: %*phN\n",
71 * Because the event buffer is located in a page which is owned
72 * by the SCLP core, all data of interest must be copied. The
73 * error indication is in 'sclp_ftp_ldflg'
75 sclp_ftp_ldflg
= diag
->mdd
.ftp
.ldflg
;
76 sclp_ftp_fsize
= diag
->mdd
.ftp
.fsize
;
77 sclp_ftp_length
= diag
->mdd
.ftp
.length
;
79 complete(&sclp_ftp_rx_complete
);
83 * sclp_ftp_et7() - start a Diagnostic Test FTP Service SCLP request
84 * @ftp: pointer to FTP descriptor
86 * Return: 0 on success, else a (negative) error code
88 static int sclp_ftp_et7(const struct hmcdrv_ftp_cmdspec
*ftp
)
90 struct completion completion
;
91 struct sclp_diag_sccb
*sccb
;
96 req
= kzalloc(sizeof(*req
), GFP_KERNEL
);
97 sccb
= (void *) get_zeroed_page(GFP_KERNEL
| GFP_DMA
);
103 sccb
->hdr
.length
= SCLP_DIAG_FTP_EVBUF_LEN
+
104 sizeof(struct sccb_header
);
105 sccb
->evbuf
.hdr
.type
= EVTYP_DIAG_TEST
;
106 sccb
->evbuf
.hdr
.length
= SCLP_DIAG_FTP_EVBUF_LEN
;
107 sccb
->evbuf
.hdr
.flags
= 0; /* clear processed-buffer */
108 sccb
->evbuf
.route
= SCLP_DIAG_FTP_ROUTE
;
109 sccb
->evbuf
.mdd
.ftp
.pcx
= SCLP_DIAG_FTP_XPCX
;
110 sccb
->evbuf
.mdd
.ftp
.srcflg
= 0;
111 sccb
->evbuf
.mdd
.ftp
.pgsize
= 0;
112 sccb
->evbuf
.mdd
.ftp
.asce
= _ASCE_REAL_SPACE
;
113 sccb
->evbuf
.mdd
.ftp
.ldflg
= SCLP_DIAG_FTP_LDFAIL
;
114 sccb
->evbuf
.mdd
.ftp
.fsize
= 0;
115 sccb
->evbuf
.mdd
.ftp
.cmd
= ftp
->id
;
116 sccb
->evbuf
.mdd
.ftp
.offset
= ftp
->ofs
;
117 sccb
->evbuf
.mdd
.ftp
.length
= ftp
->len
;
118 sccb
->evbuf
.mdd
.ftp
.bufaddr
= virt_to_phys(ftp
->buf
);
120 len
= strscpy(sccb
->evbuf
.mdd
.ftp
.fident
, ftp
->fname
,
121 HMCDRV_FTP_FIDENT_MAX
);
127 req
->command
= SCLP_CMDW_WRITE_EVENT_DATA
;
129 req
->status
= SCLP_REQ_FILLED
;
130 req
->callback
= sclp_ftp_txcb
;
131 req
->callback_data
= &completion
;
133 init_completion(&completion
);
135 rc
= sclp_add_request(req
);
139 /* Wait for end of ftp sclp command. */
140 wait_for_completion(&completion
);
143 pr_debug("status of SCLP (ET7) request is 0x%04x (0x%02x)\n",
144 sccb
->hdr
.response_code
, sccb
->evbuf
.hdr
.flags
);
148 * Check if sclp accepted the request. The data transfer runs
149 * asynchronously and the completion is indicated with an
152 if (req
->status
!= SCLP_REQ_DONE
||
153 (sccb
->evbuf
.hdr
.flags
& 0x80) == 0 || /* processed-buffer */
154 (sccb
->hdr
.response_code
& 0xffU
) != 0x20U
) {
159 free_page((unsigned long) sccb
);
165 * sclp_ftp_cmd() - executes a HMC related SCLP Diagnose (ET7) FTP command
166 * @ftp: pointer to FTP command specification
167 * @fsize: return of file size (or NULL if undesirable)
169 * Attention: Notice that this function is not reentrant - so the caller
170 * must ensure locking.
172 * Return: number of bytes read/written or a (negative) error code
174 ssize_t
sclp_ftp_cmd(const struct hmcdrv_ftp_cmdspec
*ftp
, size_t *fsize
)
178 unsigned long start_jiffies
;
180 pr_debug("starting SCLP (ET7), cmd %d for '%s' at %lld with %zd bytes\n",
181 ftp
->id
, ftp
->fname
, (long long) ftp
->ofs
, ftp
->len
);
182 start_jiffies
= jiffies
;
185 init_completion(&sclp_ftp_rx_complete
);
187 /* Start ftp sclp command. */
188 len
= sclp_ftp_et7(ftp
);
193 * There is no way to cancel the sclp ET7 request, the code
194 * needs to wait unconditionally until the transfer is complete.
196 wait_for_completion(&sclp_ftp_rx_complete
);
199 pr_debug("completed SCLP (ET7) request after %lu ms (all)\n",
200 (jiffies
- start_jiffies
) * 1000 / HZ
);
201 pr_debug("return code of SCLP (ET7) FTP Service is 0x%02x, with %lld/%lld bytes\n",
202 sclp_ftp_ldflg
, sclp_ftp_length
, sclp_ftp_fsize
);
205 switch (sclp_ftp_ldflg
) {
206 case SCLP_DIAG_FTP_OK
:
207 len
= sclp_ftp_length
;
209 *fsize
= sclp_ftp_fsize
;
211 case SCLP_DIAG_FTP_LDNPERM
:
214 case SCLP_DIAG_FTP_LDRUNS
:
217 case SCLP_DIAG_FTP_LDFAIL
:
232 static struct sclp_register sclp_ftp_event
= {
233 .send_mask
= EVTYP_DIAG_TEST_MASK
, /* want tx events */
234 .receive_mask
= EVTYP_DIAG_TEST_MASK
, /* want rx events */
235 .receiver_fn
= sclp_ftp_rxcb
, /* async callback (rx) */
236 .state_change_fn
= NULL
,
240 * sclp_ftp_startup() - startup of FTP services, when running on LPAR
242 int sclp_ftp_startup(void)
249 rc
= sclp_register(&sclp_ftp_event
);
254 info
= get_zeroed_page(GFP_KERNEL
);
257 struct sysinfo_2_2_2
*info222
= (struct sysinfo_2_2_2
*)info
;
259 if (!stsi(info222
, 2, 2, 2)) { /* get SYSIB 2.2.2 */
260 info222
->name
[sizeof(info222
->name
) - 1] = '\0';
261 EBCASC_500(info222
->name
, sizeof(info222
->name
) - 1);
262 pr_debug("SCLP (ET7) FTP Service working on LPAR %u (%s)\n",
263 info222
->lpar_number
, info222
->name
);
273 * sclp_ftp_shutdown() - shutdown of FTP services, when running on LPAR
275 void sclp_ftp_shutdown(void)
277 sclp_unregister(&sclp_ftp_event
);