1*1c60b9acSAndroid Build Coastguard Worker /*
2*1c60b9acSAndroid Build Coastguard Worker * SPI bitbang implementation using generic gpio
3*1c60b9acSAndroid Build Coastguard Worker *
4*1c60b9acSAndroid Build Coastguard Worker * Copyright (C) 2019 - 2020 Andy Green <[email protected]>
5*1c60b9acSAndroid Build Coastguard Worker *
6*1c60b9acSAndroid Build Coastguard Worker * Permission is hereby granted, free of charge, to any person obtaining a copy
7*1c60b9acSAndroid Build Coastguard Worker * of this software and associated documentation files (the "Software"), to
8*1c60b9acSAndroid Build Coastguard Worker * deal in the Software without restriction, including without limitation the
9*1c60b9acSAndroid Build Coastguard Worker * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or
10*1c60b9acSAndroid Build Coastguard Worker * sell copies of the Software, and to permit persons to whom the Software is
11*1c60b9acSAndroid Build Coastguard Worker * furnished to do so, subject to the following conditions:
12*1c60b9acSAndroid Build Coastguard Worker *
13*1c60b9acSAndroid Build Coastguard Worker * The above copyright notice and this permission notice shall be included in
14*1c60b9acSAndroid Build Coastguard Worker * all copies or substantial portions of the Software.
15*1c60b9acSAndroid Build Coastguard Worker *
16*1c60b9acSAndroid Build Coastguard Worker * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
17*1c60b9acSAndroid Build Coastguard Worker * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
18*1c60b9acSAndroid Build Coastguard Worker * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
19*1c60b9acSAndroid Build Coastguard Worker * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
20*1c60b9acSAndroid Build Coastguard Worker * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
21*1c60b9acSAndroid Build Coastguard Worker * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
22*1c60b9acSAndroid Build Coastguard Worker * IN THE SOFTWARE.
23*1c60b9acSAndroid Build Coastguard Worker */
24*1c60b9acSAndroid Build Coastguard Worker #include <libwebsockets.h>
25*1c60b9acSAndroid Build Coastguard Worker
26*1c60b9acSAndroid Build Coastguard Worker int
lws_bb_spi_init(const lws_spi_ops_t * octx)27*1c60b9acSAndroid Build Coastguard Worker lws_bb_spi_init(const lws_spi_ops_t *octx)
28*1c60b9acSAndroid Build Coastguard Worker {
29*1c60b9acSAndroid Build Coastguard Worker lws_bb_spi_t *ctx = (lws_bb_spi_t *)octx;
30*1c60b9acSAndroid Build Coastguard Worker int n;
31*1c60b9acSAndroid Build Coastguard Worker
32*1c60b9acSAndroid Build Coastguard Worker for (n = 0; n < LWS_SPI_BB_MAX_CH; n++) {
33*1c60b9acSAndroid Build Coastguard Worker if (ctx->flags & (1 << n))
34*1c60b9acSAndroid Build Coastguard Worker ctx->gpio->mode(ctx->ncs[n], LWSGGPIO_FL_WRITE);
35*1c60b9acSAndroid Build Coastguard Worker if (ctx->flags & (1 << (n + 4)))
36*1c60b9acSAndroid Build Coastguard Worker ctx->gpio->mode(ctx->ncmd[n], LWSGGPIO_FL_WRITE);
37*1c60b9acSAndroid Build Coastguard Worker }
38*1c60b9acSAndroid Build Coastguard Worker
39*1c60b9acSAndroid Build Coastguard Worker ctx->gpio->mode(ctx->clk, LWSGGPIO_FL_WRITE |
40*1c60b9acSAndroid Build Coastguard Worker ((octx->bus_mode & LWSSPIMODE_CPOL) ?
41*1c60b9acSAndroid Build Coastguard Worker 0 : LWSGGPIO_FL_START_LOW));
42*1c60b9acSAndroid Build Coastguard Worker ctx->gpio->mode(ctx->mosi, LWSGGPIO_FL_WRITE | LWSGGPIO_FL_START_LOW);
43*1c60b9acSAndroid Build Coastguard Worker ctx->gpio->mode(ctx->miso, LWSGGPIO_FL_READ | LWSGGPIO_FL_PULLUP);
44*1c60b9acSAndroid Build Coastguard Worker
45*1c60b9acSAndroid Build Coastguard Worker return 0;
46*1c60b9acSAndroid Build Coastguard Worker }
47*1c60b9acSAndroid Build Coastguard Worker
48*1c60b9acSAndroid Build Coastguard Worker /* if active, prepare DnC before this and call separately for Cmd / Data */
49*1c60b9acSAndroid Build Coastguard Worker
50*1c60b9acSAndroid Build Coastguard Worker static void
lws_bb_spi_write(lws_bb_spi_t * ctx,const uint8_t * buf,size_t len)51*1c60b9acSAndroid Build Coastguard Worker lws_bb_spi_write(lws_bb_spi_t *ctx, const uint8_t *buf, size_t len)
52*1c60b9acSAndroid Build Coastguard Worker {
53*1c60b9acSAndroid Build Coastguard Worker uint8_t u, inv = !!(ctx->bb_ops.bus_mode & LWSSPIMODE_CPOL);
54*1c60b9acSAndroid Build Coastguard Worker
55*1c60b9acSAndroid Build Coastguard Worker while (len--) {
56*1c60b9acSAndroid Build Coastguard Worker int n;
57*1c60b9acSAndroid Build Coastguard Worker
58*1c60b9acSAndroid Build Coastguard Worker u = *buf++;
59*1c60b9acSAndroid Build Coastguard Worker
60*1c60b9acSAndroid Build Coastguard Worker for (n = 0; n < 4; n++) {
61*1c60b9acSAndroid Build Coastguard Worker ctx->gpio->set(ctx->clk, inv);
62*1c60b9acSAndroid Build Coastguard Worker ctx->gpio->set(ctx->mosi, !!(u & 0x80));
63*1c60b9acSAndroid Build Coastguard Worker ctx->gpio->set(ctx->clk, !inv);
64*1c60b9acSAndroid Build Coastguard Worker ctx->gpio->set(ctx->clk, inv);
65*1c60b9acSAndroid Build Coastguard Worker ctx->gpio->set(ctx->mosi, !!(u & 0x40));
66*1c60b9acSAndroid Build Coastguard Worker ctx->gpio->set(ctx->clk, !inv);
67*1c60b9acSAndroid Build Coastguard Worker u <<= 2;
68*1c60b9acSAndroid Build Coastguard Worker }
69*1c60b9acSAndroid Build Coastguard Worker }
70*1c60b9acSAndroid Build Coastguard Worker
71*1c60b9acSAndroid Build Coastguard Worker ctx->gpio->set(ctx->clk, 0 ^ inv);
72*1c60b9acSAndroid Build Coastguard Worker }
73*1c60b9acSAndroid Build Coastguard Worker
74*1c60b9acSAndroid Build Coastguard Worker static void
lws_bb_spi_read(lws_bb_spi_t * ctx,uint8_t * buf,size_t len)75*1c60b9acSAndroid Build Coastguard Worker lws_bb_spi_read(lws_bb_spi_t *ctx, uint8_t *buf, size_t len)
76*1c60b9acSAndroid Build Coastguard Worker {
77*1c60b9acSAndroid Build Coastguard Worker uint8_t u = 0;
78*1c60b9acSAndroid Build Coastguard Worker uint8_t inv = !!(ctx->bb_ops.bus_mode & LWSSPIMODE_CPOL);
79*1c60b9acSAndroid Build Coastguard Worker
80*1c60b9acSAndroid Build Coastguard Worker while (len--) {
81*1c60b9acSAndroid Build Coastguard Worker int n;
82*1c60b9acSAndroid Build Coastguard Worker
83*1c60b9acSAndroid Build Coastguard Worker for (n = 0; n < 8; n++) {
84*1c60b9acSAndroid Build Coastguard Worker ctx->gpio->set(ctx->clk, inv);
85*1c60b9acSAndroid Build Coastguard Worker u = (u << 1) | !!ctx->gpio->read(ctx->miso);
86*1c60b9acSAndroid Build Coastguard Worker ctx->gpio->set(ctx->mosi, !!(u & 0x80));
87*1c60b9acSAndroid Build Coastguard Worker ctx->gpio->set(ctx->clk, !inv);
88*1c60b9acSAndroid Build Coastguard Worker }
89*1c60b9acSAndroid Build Coastguard Worker *buf++ = u;
90*1c60b9acSAndroid Build Coastguard Worker }
91*1c60b9acSAndroid Build Coastguard Worker
92*1c60b9acSAndroid Build Coastguard Worker ctx->gpio->set(ctx->clk, 0 ^ inv);
93*1c60b9acSAndroid Build Coastguard Worker }
94*1c60b9acSAndroid Build Coastguard Worker
95*1c60b9acSAndroid Build Coastguard Worker int
lws_bb_spi_queue(const lws_spi_ops_t * octx,const lws_spi_desc_t * desc)96*1c60b9acSAndroid Build Coastguard Worker lws_bb_spi_queue(const lws_spi_ops_t *octx, const lws_spi_desc_t *desc)
97*1c60b9acSAndroid Build Coastguard Worker {
98*1c60b9acSAndroid Build Coastguard Worker lws_bb_spi_t *ctx = (lws_bb_spi_t *)octx;
99*1c60b9acSAndroid Build Coastguard Worker const uint8_t *src = desc->src;
100*1c60b9acSAndroid Build Coastguard Worker
101*1c60b9acSAndroid Build Coastguard Worker /* clock to idle */
102*1c60b9acSAndroid Build Coastguard Worker ctx->gpio->set(ctx->clk, 0 ^ !!(octx->bus_mode & LWSSPIMODE_CPOL));
103*1c60b9acSAndroid Build Coastguard Worker /* enable nCS */
104*1c60b9acSAndroid Build Coastguard Worker ctx->gpio->set(ctx->ncs[desc->channel], 0);
105*1c60b9acSAndroid Build Coastguard Worker
106*1c60b9acSAndroid Build Coastguard Worker if (desc->count_cmd) {
107*1c60b9acSAndroid Build Coastguard Worker ctx->gpio->set(ctx->ncmd[desc->channel], 0);
108*1c60b9acSAndroid Build Coastguard Worker lws_bb_spi_write(ctx, src, desc->count_cmd);
109*1c60b9acSAndroid Build Coastguard Worker ctx->gpio->set(ctx->ncmd[desc->channel], 1);
110*1c60b9acSAndroid Build Coastguard Worker
111*1c60b9acSAndroid Build Coastguard Worker src += desc->count_cmd;
112*1c60b9acSAndroid Build Coastguard Worker }
113*1c60b9acSAndroid Build Coastguard Worker
114*1c60b9acSAndroid Build Coastguard Worker if (desc->count_write)
115*1c60b9acSAndroid Build Coastguard Worker lws_bb_spi_write(ctx, desc->data, desc->count_write);
116*1c60b9acSAndroid Build Coastguard Worker
117*1c60b9acSAndroid Build Coastguard Worker if (desc->count_read)
118*1c60b9acSAndroid Build Coastguard Worker lws_bb_spi_read(ctx, desc->dest, desc->count_read);
119*1c60b9acSAndroid Build Coastguard Worker
120*1c60b9acSAndroid Build Coastguard Worker /* disable nCS */
121*1c60b9acSAndroid Build Coastguard Worker ctx->gpio->set(ctx->ncs[desc->channel], 1);
122*1c60b9acSAndroid Build Coastguard Worker
123*1c60b9acSAndroid Build Coastguard Worker /* clock to idle */
124*1c60b9acSAndroid Build Coastguard Worker ctx->gpio->set(ctx->clk, 0 ^ !!(octx->bus_mode & LWSSPIMODE_CPOL));
125*1c60b9acSAndroid Build Coastguard Worker
126*1c60b9acSAndroid Build Coastguard Worker return 0;
127*1c60b9acSAndroid Build Coastguard Worker }
128