1*c82ee6d3SThomas Gleixner /* SPDX-License-Identifier: GPL-2.0-or-later */
2c6fd2807SJeff Garzik /*
3c6fd2807SJeff Garzik * sata_promise.h - Promise SATA common definitions and inline funcs
4c6fd2807SJeff Garzik *
5c6fd2807SJeff Garzik * Copyright 2003-2004 Red Hat, Inc.
6c6fd2807SJeff Garzik *
7c6fd2807SJeff Garzik * libata documentation is available via 'make {ps|pdf}docs',
819285f3cSMauro Carvalho Chehab * as Documentation/driver-api/libata.rst
9c6fd2807SJeff Garzik */
10c6fd2807SJeff Garzik
11c6fd2807SJeff Garzik #ifndef __SATA_PROMISE_H__
12c6fd2807SJeff Garzik #define __SATA_PROMISE_H__
13c6fd2807SJeff Garzik
14c6fd2807SJeff Garzik #include <linux/ata.h>
15c6fd2807SJeff Garzik
16c6fd2807SJeff Garzik enum pdc_packet_bits {
17c6fd2807SJeff Garzik PDC_PKT_READ = (1 << 2),
18c6fd2807SJeff Garzik PDC_PKT_NODATA = (1 << 3),
19c6fd2807SJeff Garzik
20c6fd2807SJeff Garzik PDC_PKT_SIZEMASK = (1 << 7) | (1 << 6) | (1 << 5),
21c6fd2807SJeff Garzik PDC_PKT_CLEAR_BSY = (1 << 4),
22c6fd2807SJeff Garzik PDC_PKT_WAIT_DRDY = (1 << 3) | (1 << 4),
23c6fd2807SJeff Garzik PDC_LAST_REG = (1 << 3),
24c6fd2807SJeff Garzik
25c6fd2807SJeff Garzik PDC_REG_DEVCTL = (1 << 3) | (1 << 2) | (1 << 1),
26c6fd2807SJeff Garzik };
27c6fd2807SJeff Garzik
pdc_pkt_header(struct ata_taskfile * tf,dma_addr_t sg_table,unsigned int devno,u8 * buf)28c6fd2807SJeff Garzik static inline unsigned int pdc_pkt_header(struct ata_taskfile *tf,
29c6fd2807SJeff Garzik dma_addr_t sg_table,
30c6fd2807SJeff Garzik unsigned int devno, u8 *buf)
31c6fd2807SJeff Garzik {
32c6fd2807SJeff Garzik u8 dev_reg;
334ca4e439SAl Viro __le32 *buf32 = (__le32 *) buf;
34c6fd2807SJeff Garzik
35c6fd2807SJeff Garzik /* set control bits (byte 0), zero delay seq id (byte 3),
36c6fd2807SJeff Garzik * and seq id (byte 2)
37c6fd2807SJeff Garzik */
38c6fd2807SJeff Garzik switch (tf->protocol) {
39c6fd2807SJeff Garzik case ATA_PROT_DMA:
40c6fd2807SJeff Garzik if (!(tf->flags & ATA_TFLAG_WRITE))
41c6fd2807SJeff Garzik buf32[0] = cpu_to_le32(PDC_PKT_READ);
42c6fd2807SJeff Garzik else
43c6fd2807SJeff Garzik buf32[0] = 0;
44c6fd2807SJeff Garzik break;
45c6fd2807SJeff Garzik
46c6fd2807SJeff Garzik case ATA_PROT_NODATA:
47c6fd2807SJeff Garzik buf32[0] = cpu_to_le32(PDC_PKT_NODATA);
48c6fd2807SJeff Garzik break;
49c6fd2807SJeff Garzik
50c6fd2807SJeff Garzik default:
51c6fd2807SJeff Garzik BUG();
52c6fd2807SJeff Garzik break;
53c6fd2807SJeff Garzik }
54c6fd2807SJeff Garzik
55c6fd2807SJeff Garzik buf32[1] = cpu_to_le32(sg_table); /* S/G table addr */
56c6fd2807SJeff Garzik buf32[2] = 0; /* no next-packet */
57c6fd2807SJeff Garzik
58c6fd2807SJeff Garzik if (devno == 0)
59c6fd2807SJeff Garzik dev_reg = ATA_DEVICE_OBS;
60c6fd2807SJeff Garzik else
61c6fd2807SJeff Garzik dev_reg = ATA_DEVICE_OBS | ATA_DEV1;
62c6fd2807SJeff Garzik
63c6fd2807SJeff Garzik /* select device */
64c6fd2807SJeff Garzik buf[12] = (1 << 5) | PDC_PKT_CLEAR_BSY | ATA_REG_DEVICE;
65c6fd2807SJeff Garzik buf[13] = dev_reg;
66c6fd2807SJeff Garzik
67c6fd2807SJeff Garzik /* device control register */
68c6fd2807SJeff Garzik buf[14] = (1 << 5) | PDC_REG_DEVCTL;
69c6fd2807SJeff Garzik buf[15] = tf->ctl;
70c6fd2807SJeff Garzik
71c6fd2807SJeff Garzik return 16; /* offset of next byte */
72c6fd2807SJeff Garzik }
73c6fd2807SJeff Garzik
pdc_pkt_footer(struct ata_taskfile * tf,u8 * buf,unsigned int i)74c6fd2807SJeff Garzik static inline unsigned int pdc_pkt_footer(struct ata_taskfile *tf, u8 *buf,
75c6fd2807SJeff Garzik unsigned int i)
76c6fd2807SJeff Garzik {
77c6fd2807SJeff Garzik if (tf->flags & ATA_TFLAG_DEVICE) {
78c6fd2807SJeff Garzik buf[i++] = (1 << 5) | ATA_REG_DEVICE;
79c6fd2807SJeff Garzik buf[i++] = tf->device;
80c6fd2807SJeff Garzik }
81c6fd2807SJeff Garzik
82c6fd2807SJeff Garzik /* and finally the command itself; also includes end-of-pkt marker */
83c6fd2807SJeff Garzik buf[i++] = (1 << 5) | PDC_LAST_REG | ATA_REG_CMD;
84c6fd2807SJeff Garzik buf[i++] = tf->command;
85c6fd2807SJeff Garzik
86c6fd2807SJeff Garzik return i;
87c6fd2807SJeff Garzik }
88c6fd2807SJeff Garzik
pdc_prep_lba28(struct ata_taskfile * tf,u8 * buf,unsigned int i)89c6fd2807SJeff Garzik static inline unsigned int pdc_prep_lba28(struct ata_taskfile *tf, u8 *buf, unsigned int i)
90c6fd2807SJeff Garzik {
91c6fd2807SJeff Garzik /* the "(1 << 5)" should be read "(count << 5)" */
92c6fd2807SJeff Garzik
93c6fd2807SJeff Garzik /* ATA command block registers */
94c6fd2807SJeff Garzik buf[i++] = (1 << 5) | ATA_REG_FEATURE;
95c6fd2807SJeff Garzik buf[i++] = tf->feature;
96c6fd2807SJeff Garzik
97c6fd2807SJeff Garzik buf[i++] = (1 << 5) | ATA_REG_NSECT;
98c6fd2807SJeff Garzik buf[i++] = tf->nsect;
99c6fd2807SJeff Garzik
100c6fd2807SJeff Garzik buf[i++] = (1 << 5) | ATA_REG_LBAL;
101c6fd2807SJeff Garzik buf[i++] = tf->lbal;
102c6fd2807SJeff Garzik
103c6fd2807SJeff Garzik buf[i++] = (1 << 5) | ATA_REG_LBAM;
104c6fd2807SJeff Garzik buf[i++] = tf->lbam;
105c6fd2807SJeff Garzik
106c6fd2807SJeff Garzik buf[i++] = (1 << 5) | ATA_REG_LBAH;
107c6fd2807SJeff Garzik buf[i++] = tf->lbah;
108c6fd2807SJeff Garzik
109c6fd2807SJeff Garzik return i;
110c6fd2807SJeff Garzik }
111c6fd2807SJeff Garzik
pdc_prep_lba48(struct ata_taskfile * tf,u8 * buf,unsigned int i)112c6fd2807SJeff Garzik static inline unsigned int pdc_prep_lba48(struct ata_taskfile *tf, u8 *buf, unsigned int i)
113c6fd2807SJeff Garzik {
114c6fd2807SJeff Garzik /* the "(2 << 5)" should be read "(count << 5)" */
115c6fd2807SJeff Garzik
116c6fd2807SJeff Garzik /* ATA command block registers */
117c6fd2807SJeff Garzik buf[i++] = (2 << 5) | ATA_REG_FEATURE;
118c6fd2807SJeff Garzik buf[i++] = tf->hob_feature;
119c6fd2807SJeff Garzik buf[i++] = tf->feature;
120c6fd2807SJeff Garzik
121c6fd2807SJeff Garzik buf[i++] = (2 << 5) | ATA_REG_NSECT;
122c6fd2807SJeff Garzik buf[i++] = tf->hob_nsect;
123c6fd2807SJeff Garzik buf[i++] = tf->nsect;
124c6fd2807SJeff Garzik
125c6fd2807SJeff Garzik buf[i++] = (2 << 5) | ATA_REG_LBAL;
126c6fd2807SJeff Garzik buf[i++] = tf->hob_lbal;
127c6fd2807SJeff Garzik buf[i++] = tf->lbal;
128c6fd2807SJeff Garzik
129c6fd2807SJeff Garzik buf[i++] = (2 << 5) | ATA_REG_LBAM;
130c6fd2807SJeff Garzik buf[i++] = tf->hob_lbam;
131c6fd2807SJeff Garzik buf[i++] = tf->lbam;
132c6fd2807SJeff Garzik
133c6fd2807SJeff Garzik buf[i++] = (2 << 5) | ATA_REG_LBAH;
134c6fd2807SJeff Garzik buf[i++] = tf->hob_lbah;
135c6fd2807SJeff Garzik buf[i++] = tf->lbah;
136c6fd2807SJeff Garzik
137c6fd2807SJeff Garzik return i;
138c6fd2807SJeff Garzik }
139c6fd2807SJeff Garzik
140c6fd2807SJeff Garzik
141c6fd2807SJeff Garzik #endif /* __SATA_PROMISE_H__ */
142