Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

ESP32 and ESP 8266 #152

Open
jpitz31 opened this issue Jan 31, 2024 · 55 comments
Open

ESP32 and ESP 8266 #152

jpitz31 opened this issue Jan 31, 2024 · 55 comments

Comments

@jpitz31
Copy link

jpitz31 commented Jan 31, 2024

Just wondering if you have thought of porting rfcat to run on ESP32 and ESP8266, These processors are readily available and very inexpensive. Just add a CC1101 or a Nrf905.

Thanks
Joe

@atlas0fd00m
Copy link
Owner

i haven't, actually. i'm not opposed. would you like to take the first swing at a hardware/firmware combination that could work? i'd be happy to help guide you through the process. any interest?

@jpitz31
Copy link
Author

jpitz31 commented Jan 31, 2024

I have been thinking about it. I am not familiar with the CC1111, but familiar with the ESP series. Let me take a look at the source. If I can get a grasp of what is involved I will let you know. Thanks for the offer.

Joe

@jpitz31
Copy link
Author

jpitz31 commented Feb 2, 2024

What platform are you developing with? Visual Studio Code with C/C++ plugins or Visual Studio Code with PlatformIO, Just wondering how much of the existing firmware code could be cross compiled as it is currently. I might try that first.
Thanks
Joe

@atlas0fd00m
Copy link
Owner

currently we build with SDCC, which is pretty nonspecific.

the firmware was written specifically for the CC1111, an SOC which incorporates an 8051 micro with a USB controller and the CC1101 radio (operating at 24MHz instead of 26MHz because that's in harmony with the 48MHz clock used by the USB controller and easier for TI to build).

the parts you'll be able to (or even want to) reuse have to do with the command-protocol. that's pretty simple.
the rest of the firmware is basically just being a go-between for the Radio and the USB controller. honestly, the hardest part about RfCat firmware development was because i wrote the USB stack from scratch. i'm certain that the ESP* devkits will include whatever you need for USB and/or WiFi to make it easy :)

the CC1101 and the CC1111 are virtually identical from an interface perspective. you'll access the CC1101 differently, since it will be an external device. but once that tiny shim layer is done, the registers that control the radio are all the same. the 26/24MHz thing is just a calculation difference for the numbers that get put into certain RF control registers (and when using the Python code, it's already baked into the code to allow for either one). at one point i wanted to support the IMME dongle (which had a CC1101), but it's proprietary firmware/design and cypress chip were ultimately more than i had time or patience for.

i hope this helps.
good luck!
@

@atlas0fd00m
Copy link
Owner

oh, and i wrote it all in VIM. i'm pretty sure if VSCode is you thing, it will go well for you. i've considered it myself.

@Crsarmv7l
Copy link
Contributor

Crsarmv7l commented Feb 3, 2024

i haven't, actually. i'm not opposed. would you like to take the first swing at a hardware/firmware combination that could work? i'd be happy to help guide you through the process. any interest?

Lol, I am actually working on such, slowly (work and kids mean I get to it only for an hour or so each weekend). Its an ugly combo, essentially reading serial input with RfCat style commands, which translate to the appropriate Elechouse lib for cc1101 control. Nice thing is I hopefully will incorporate a few decoders as well. rx on core 0, decode on core 1. Also cheap hardware plus the ability to automate with pyserial and it can work with any device.

Got the parsing done, need to modify the Elechouse lib though for quite a few issues.

@Crsarmv7l
Copy link
Contributor

Crsarmv7l commented Feb 3, 2024

BTW @jpitz31 As atlas said the registers are very similar with cc1101 and cc1111 but there are differences (eg register addresses etc). By far the biggest difference is the cc1101 has a FIFO buffer which the cc1111 doesn't but there are a few other differences here and there with the registers.

My first whack with porting to esp/cc1101 I had hoped to use the chipcon_nic.py functions with modified registers/crystal at 26mhz etc along with the fifo buffer since python is slow with micropython. That didn't work out very well (although it did work) so now I am doing it appropriately with C++. My C++ isn't the best though and I am moving slow on it so feel free to attempt!

@jpitz31
Copy link
Author

jpitz31 commented Feb 5, 2024

Thanks @atlas0fd00m and @Crsarmv7l I will start looking over the firmware code.

@jpitz31
Copy link
Author

jpitz31 commented Feb 5, 2024

Yes, on linix firmware is loaded using esptool.py and is pretty easy to perform. On windows there is a Espressif app that allows users to flash the firmware. At first glance it appears that the firmware handles USB, bluetooth and wifi. To do anything, library functions or methods are used to communicate between the different devices. I have been playing around with RadioLib to communicate with CC1101 radios, pretty easy methods to call. Maybe RadioLib can be used as a model for radio communication. But I am still looking over the code. I also want to get familiar with how the client tools interact with the firmware, I want to try to leave all of the client stuff alone.

Thanks
Joe

@jpitz31
Copy link
Author

jpitz31 commented Feb 6, 2024

Here is a link to ESP-IDF (Espressif IoT Development Framework), https://docs.espressif.com/projects/esp-idf/en/latest/esp32/get-started/ There is a Visual Studio plugin for this toolchain.

As I mentioned earlier, I am looking at Radiolib, several other RF projects use this lib, It supports 12 different radios and has an abstract layer to add other platforms. RFQuack uses Radiolib.

I am going to setup the toolchain and see how it works. Also trying to locate some docs on how the firmware communicates with the onboard devices ie: USB, BLE and Wifi. Will post my findings.

ALso looking at CMake and Ninja, it is cross platform, works on Windows, linux and Mac.

Thanks
Joe

@jpitz31
Copy link
Author

jpitz31 commented Feb 6, 2024

Here is the tutorial I followed to setup the ESP-IDF
https://www.espboards.dev/blog/use-esp-idf-with-vscode/

@jpitz31
Copy link
Author

jpitz31 commented Feb 7, 2024

Hey, Atlas, In the sample programs that came with the ESP-IDF tool chain, there is a CLI example program that allows you to query chip related info. This is sent directly through USB to the ESP32.
I have been going through the code for rfcat. I see there are several .c files that start with app*.c and one application.c file.
Could you give me a quick run down on the program flow of files for the firmware for the C1111 radio. I also notice that sections of code are commented out, // not used by VCOM. Could you give me some background on this as well. What is the flow of how the firmware parses the in-comming python commands, appears to be stored in the ep5.OUTbuf.

Thanks

Joe

@atlas0fd00m
Copy link
Owner

sure!

so, the cc1111* files are mostly related to the RF part of the chip (all except the cc1111_vcom* files, which i apparently let slip through the pattern buffers unchecked).

the chipcon_* files are more about the details of making the microcontroller work. primarily chipcon_usb* the the code that manages the USB controller, and subsequently has a lot of the core functionality in it. (the cc1111_vcom* files are somewhat out-of-date replacements for chipcon_usb* written by someone way smarter than me, who decided using a more standard USB stack to make the dongle show up as a TTY/COM port was a better idea. but no! i had already spent so much time learning and making the USB stack from scratch, and i liked being able to twiddle the USB-bits to make hidden magic happen... so i continued to focus on my own code instead of the probably better vcom code)

the app* files are the glue that binds them together into one cohesive firmware. (notice, they're the only files with main() functions :)

and the imme* files are all completely different, but you can consider them a sort of app* set of files, intended to create awesome apps on the IMME girl-toys.

as far as the data getting to ep5.OUTbuf, the USB firmware initializes USB endpoint 5 (ep5, as defined in the USB docs for this chip). because of the standard verbology of USB, the buffers are named with regard to the Master (the computer), so it's an OUT buffer of data, and thus the name OUTbuf :) the characteristics of EP5 are initialized in ``

USB processing for EP5 starts with the Interrupt Handler usbIntHandler() which is initialized as the P2 Interrupt Vector handler, and basically just grabs the interrupt event flags and stores them in the usb_data.event field, does a little housekeeping, and returns.

you'll notice in usbProcessEvents(), after handling power state and EP0 stuff, it checks to see if usb_data.event has the USBD_OIF_OUTEP5IF flag set, indicating data is in the EP5 OUT buffer and needs to be handled. if so, it calls handleOUTEP5() and checks the results.

handleOUTEP5() basically grabs the message metadata and sticks it in ep5.*, then sets up DMA for copying the data from the USB buffer to the global memory buffer ep5.OUTbuf. when all that's done, it returns and usbProcessEvents() calls processOUTEP5() to actually process the data in the buffer.

processOUTEP5() is the first higher level function in the stack that actually deals with the data you're sending the RfCat over USB.

so you have this buffer, and you don't know what to do with it. so you make a simple protocol.
you think to yourself, "hey self. you might have multiple apps running on this thing at once, and each app probably wants to be able to have a bunch of commands (or "verbs").
so you make the first two bytes of every message <app> and <cmd>.
you later think "cool! i'm glad i did that cuz i can have this cool app called.... ahem..... let there be a SYSTEM app." and boom, there was a "SYSTEM" app, and it's app number was made 0xff. and you say to yourself, "self, this is good. now we shall be able to tell the god-like functionality from the mortal, errr. application level functionality."
ok, so it was more that the SYSTEM app stuff was stuff i just always wanted to be wrapped into every app, a sort of "bonus" with functionality like printing log messages to the Python app, and allowing the Python app to peek and poke at raw memory locations, etc.... in fact, many of the RF settings in the Python side use peek/poke to work with the radio... so you may want to rewrite those to direct some chunk of memory to directly access the registers in the CC1101.

so, the protocol is pretty simple. the biggest thing i didn't tell you yet is that, whenever a command comes in, it should be acknowledged with a txdata() call, like this:

   txdata(ep5.OUTapp, ep5.OUTcmd, ep5.OUTbytesleft, ptr); 

the app/cmd have to match, or the Python side will keep waiting for it. it's a way to return some data as well. notice the clock and the ping responses:

            case CMD_GET_CLOCK:
                txdata(ep5.OUTapp, ep5.OUTcmd, 4, (__xdata u8*)clock);
                break;
            case CMD_PING:
                blink(2,2);
                txdata(ep5.OUTapp,ep5.OUTcmd,ep5.OUTlen,ptr);
                break;

txdata() takes 4 arguments: app, cmd, length, data)

here ends the lesson for the day. i hope it's been entertaining and useful. :)

@jpitz31
Copy link
Author

jpitz31 commented Feb 10, 2024

@atlas0fd00m Thanks for the run through. I have the toolchain working and have built the console cli using an example provided by ESP-IDF. To support many different radios. I have figured out a way to use a config.ini file to have cmake include different .c files and plan to write different .c files for each radio. So all you have to do is set the config.ini to the radio you want to use and then cmake the project. Radiolib supports 12 radios and different platforms.

The example uses a terminal console that uses serial communication. The ESP-IDF provides python code to launch the terminal console. This would be for the interactive CLI. Since the ESP32 is listening all of the time, I can write a python script to read a script as rfcat does.

I have tested radiolib with a CC1101 and can transmit and receive just fine.

The CLI example uses pointers to a function so it should be pretty easy to call the different functions to interface with the radios.

The menu options are dynamically built including argc and argv so custom commands can be built depending on each radio argument requirements. radiolib handles the communication between the processor and the radio using the SPI interface.

The example program registers functions that are called by function pointers and I am now looking at the code to model each radio function needed to call each radio.

Thanks

Joe

@Crsarmv7l
Copy link
Contributor

Crsarmv7l commented Feb 16, 2024

The example uses a terminal console that uses serial communication. The ESP-IDF provides python code to launch the terminal console. This would be for the interactive CLI. Since the ESP32 is listening all of the time, I can write a python script to read a script as rfcat does.

Basically same thing I currently have done. ESP32 has a serial buffer which I am using to receive user entered commands (also keystop(), I love that function) and set the radio/rx/tx radio packets. Pyserial was also my plan to interact with serial input/output instead of pulling an @atlas0fd00m and writing the usb stack.

Also something I did that you might want to do is dedicate ESP32 core 0 to rx, then pass off the rxed packet to core 1 (Queueing) for print/decode etc freeing up core 0 to rx the next potential packet.

I need to do the actual radio and decode pieces with my project, been to busy the last few weeks to do anything.

Sounds like you are moving along. Keep at it

@atlas0fd00m
Copy link
Owner

@jpitz31 @Crsarmv7l , does it make sense for you two to work together on your project(s)?

it occurs to me that the python side will want to change to accommodate your new hardware. please let me know if you'd like some help matching that all together.

do you have a reference hardware combination yet? pictures or a link? :)

@

@jpitz31
Copy link
Author

jpitz31 commented Feb 23, 2024 via email

@Crsarmv7l
Copy link
Contributor

Crsarmv7l commented Feb 23, 2024 via email

@Crsarmv7l
Copy link
Contributor

Crsarmv7l commented Feb 25, 2024

@jpitz31

Elechouse lib didn't work for me (although I suppose it could also be an issue with my wiring/hw/noise, once my pcbs get here I can rule that out). I was hoping it was the easy button and while I have never used GFSK or MSK, ELECHOUSE supports that while radiolib doesn't. On the plus side, I can adapt my input/output to radiolib, so I am going that direction like you.

My current plan:
Complete:

  • Serial input complete
  • Serial output complete

Pending:

  • Adapt to radiolib since Elechouse failed me
  • RX working w/output to serial
  • TX implemented
  • Verify signal integrity both TX/RX
  • Write a pyserial frontend

I like the idea of keeping pure serial input/output as a possibility because that lends to a degree of being platform agnostic. I can open a Serial connection in a webbrowser, on my phone, on anything. As a scope creep goal, Serial over BLE would also provide an interesting way to interact with the cheap hardware.

@Crsarmv7l
Copy link
Contributor

Hmm interesting. I am now getting something on rx by bitbanging gdo0 set as input and using the cc1101 in async mode. Not a very good way though because the polling rate is far too high for the given baud and I can't definitively say the actual signal is being captured.

I will need to think on this some more. I just needed to see something being received.

@jpitz31
Copy link
Author

jpitz31 commented Feb 29, 2024

Hello All, I put together a few radios and ESP32's
cc1101

I did find some code on github that can play flipperzero raw .sub files. The lib they are using is Elechouse and this worked for me.
I modified the ESP_IDF_UART_RxTx sample to read data from the serial port and then write the same text back to the serial port.

I then wrote a python program that allows users to enter a text string and send it to the esp32. The ESP32 then reads the string and writes it back to the serial port. The python reads the text and prints it to the console.

I am just using this sample program as a proof of concept. I like how the Advanced console sample app uses function pointers to execute functions. So when I get the everything working I can then write the custom code to use the more optimized code.

The radios use simple USB to UART bridge to the ESP32 chip, nothing special needed.

When I make a bit more progress I will put the code out on github.

When debugging the flipper code did you use a wifi dev board for debugging or a jtag board. When I try to debug subghz I have lots of issues with flipperzero crashing and not going directly to the subghz app. I have tried several suggestions on the flipper discord server, but still having issues.

I have tested my radio board with both Elechouse and radio and can transmit data to a doorbell I use for testing.

@Crsarmv7l
Copy link
Contributor

@jpitz31

Sure the tx piece is easy. RX is fine too with ELECHOUSE provided you have a syncword high/low.

It is falling down when I want just raw. Think RfCat lowball(1). Likely because of the FIFO and because of the IOCFG0 pino Config (unless I directly poll GDO0).

See here for code examples and man page details regarding the configs ELECHOUSE is using.
LSatan/SmartRC-CC1101-Driver-Lib#146

@atlas0fd00m any thoughts On my observations there? I am fairly sure it is either the IOCFG0 config/the FIFO or both as I do get data when polling GDO directly. I have no problem doing that either, but I need to figure out a way to mitigate the speed I am polling at.

I have a way forward just not the time to test atm.

@jpitz31
Copy link
Author

jpitz31 commented Feb 29, 2024

@Crsarmv7l ,
Not sure if you have seen this, A windows program from Ti that helps you configure the registers of the CC line of chips.
https://www.ti.com/tool/SMARTRFTM-STUDIO

Thanks

@Crsarmv7l
Copy link
Contributor

Crsarmv7l commented Mar 2, 2024

More digging shows my assumption is correct, you cannot disable preamble/syncword and still use the FIFO. Makes sense though, and why I suspected it.

image

and another:

image

Sucks though, because I had 10 pcbs made from jlpcb, for drop in cc1101 and my esp32-S3 with the traces done....but I didn't run a trace for GDO2 🤣. So bodge wire it is

I need GDO2 for sync/async serial operations for the clock. Once this is dialed in I'll be set.

Handy Application Note for Continuous Data:
https://www.ti.com/lit/an/swra359a/swra359a.pdf?ts=1709389401895&ref_url=https%253A%252F%252Fwww.ti.com%252Fproduct%252FCC1101

Atlasofdoom made it nice and simple for us with the YS1. This is a pain lol. I'll get there though. One problem at a time. Sucks though, I am doing full time automation coding at work so I don't want to touch my own projects even when I have a bit of time.

I do like the even further understanding of what is going on here. I was over polling before because I didn't have a clock (GDO2 wasn't wired),

@Crsarmv7l
Copy link
Contributor

Looks like synchronous serial setup will do what I want, while maintaining most of the FIFO/hw functionality.

Async looks like it comes with some serious drawbacks to workaround.

@jpitz31
Copy link
Author

jpitz31 commented Mar 5, 2024

Hey guys, my schedule has gone crazy, I am going to have to back out of the project. It looks like @Crsarmv7l has a good handle on the getting the radio portion dialing in.

Good luck to you all.

Thanks

@Crsarmv7l
Copy link
Contributor

Gdo2 wired. I'll get rid of the sockets so everything fits more flush, and put the code up once I get a chance to dial in rx reading from the pins directly.
IMG_20240308_190517
IMG_20240308_190506

@atlas0fd00m
Copy link
Owner

Hello All, I put together a few radios and ESP32's cc1101

I did find some code on github that can play flipperzero raw .sub files. The lib they are using is Elechouse and this worked for me. I modified the ESP_IDF_UART_RxTx sample to read data from the serial port and then write the same text back to the serial port.

I then wrote a python program that allows users to enter a text string and send it to the esp32. The ESP32 then reads the string and writes it back to the serial port. The python reads the text and prints it to the console.

I am just using this sample program as a proof of concept. I like how the Advanced console sample app uses function pointers to execute functions. So when I get the everything working I can then write the custom code to use the more optimized code.

The radios use simple USB to UART bridge to the ESP32 chip, nothing special needed.

When I make a bit more progress I will put the code out on github.

When debugging the flipper code did you use a wifi dev board for debugging or a jtag board. When I try to debug subghz I have lots of issues with flipperzero crashing and not going directly to the subghz app. I have tried several suggestions on the flipper discord server, but still having issues.

I have tested my radio board with both Elechouse and radio and can transmit data to a doorbell I use for testing.

oooooooo pretty :)

@atlas0fd00m
Copy link
Owner

@jpitz31

Sure the tx piece is easy. RX is fine too with ELECHOUSE provided you have a syncword high/low.

It is falling down when I want just raw. Think RfCat lowball(1). Likely because of the FIFO and because of the IOCFG0 pino Config (unless I directly poll GDO0).

See here for code examples and man page details regarding the configs ELECHOUSE is using. LSatan/SmartRC-CC1101-Driver-Lib#146

@atlas0fd00m any thoughts On my observations there? I am fairly sure it is either the IOCFG0 config/the FIFO or both as I do get data when polling GDO directly. I have no problem doing that either, but I need to figure out a way to mitigate the speed I am polling at.

I have a way forward just not the time to test atm.

my apologies, i understand the "not enough time to test atm." i've been running hog-wild for the past several weeks.
if you ever don't get a response from me and want one more quickly, hit me at atlas#AT#r4780y.com and hopefully i'll see it faster.

i'm not sure what to say here. i've only read about doing RX from a cc1101 chip, or used an existing library to do it... and all that was a while before i wrote lowBall. there should obviously be a trigger to indicate that there's a packet to pick up, which shouldn't have anything to do with the lowBall()-like settings (i wouldn't believe). you're still using the modem and packetizer, you're just clearing the path for it to identify packets constantly.

i would look for a race condition in the code (clearing an interrupt and missing that there's another packet), or possibly reviewing the settings you're doing in the RF CONFIG to cause a "lowball". i haven't had time to look at any code you may have shared, but it looks like you're polling the chip. is there no way for the cc1101 to let you know it's got a packet ready (like an interrupt line).
sorry, been a long time since i looked at the 1101.

good work so far tho!

@atlas0fd00m
Copy link
Owner

@Crsarmv7l , Not sure if you have seen this, A windows program from Ti that helps you configure the registers of the CC line of chips. https://www.ti.com/tool/SMARTRFTM-STUDIO

Thanks

that's how i 'trued-in" the first initial configs, and learned about the way different settings have to work :) nice find!

@atlas0fd00m
Copy link
Owner

More digging shows my assumption is correct, you cannot disable preamble/syncword and still use the FIFO. Makes sense though, and why I suspected it.

image

and another:

image

Sucks though, because I had 10 pcbs made from jlpcb, for drop in cc1101 and my esp32-S3 with the traces done....but I didn't run a trace for GDO2 🤣. So bodge wire it is

I need GDO2 for sync/async serial operations for the clock. Once this is dialed in I'll be set.

Handy Application Note for Continuous Data: https://www.ti.com/lit/an/swra359a/swra359a.pdf?ts=1709389401895&ref_url=https%253A%252F%252Fwww.ti.com%252Fproduct%252FCC1101

Atlasofdoom made it nice and simple for us with the YS1. This is a pain lol. I'll get there though. One problem at a time. Sucks though, I am doing full time automation coding at work so I don't want to touch my own projects even when I have a bit of time.

I do like the even further understanding of what is going on here. I was over polling before because I didn't have a clock (GDO2 wasn't wired),

nice work.
thank you for the kind words :)
keep this in mind tho: i just took a dongle that was already on the market (albeit wildly undervalued) and dragged the functionality out to Python and played around with easier ways to reverse/hack RF.

you're reworking much of that, and generating new hardware, and overcoming/translating the differences! i'm really excited to see what you're doing and the outcome.

ok, so i wrote a USB stack. points for masochism! and the challenge, i guess, lol (debugging USB stack code by flashing an LED with error codes in binary was quite a trip). but you all probably would have been better served if i had used the VCOM USB stack provided by Gerard van den Bosch back in 2011 (code is still in there, although i haven't run it in a decade).

during this process, let's just say it's been a trip down memory lane... tracing me through ancient repo code stored locally... which spanned Github, BitBucket, GoogleCode... and traipsed through Git, Hg, and SVN. sadly, because the initial roots (a library called cc1111usb) was in SVN on GoogleCode, and those repos are long gone, so is the SVN log data :(

but i digress. i'm proud of the work you've done and where you're going with the project :)

hack fun!
@

@atlas0fd00m
Copy link
Owner

Hey guys, my schedule has gone crazy, I am going to have to back out of the project. It looks like @Crsarmv7l has a good handle on the getting the radio portion dialing in.

Good luck to you all.

Thanks

understood. i'm sorry. i hope you get more time and can come back to it! thank you for your contributions so far, @jpitz31 .
@

@atlas0fd00m
Copy link
Owner

Gdo2 wired. I'll get rid of the sockets so everything fits more flush, and put the code up once I get a chance to dial in rx reading from the pins directly. IMG_20240308_190517 IMG_20240308_190506

LOVING this. nice work :)

@Crsarmv7l
Copy link
Contributor

my apologies, i understand the "not enough time to test atm." i've been running hog-wild for the past several weeks. if you ever don't get a response from me and want one more quickly, hit me at atlas#AT#r4780y.com and hopefully i'll see it faster.

Much appreciated with the POC info. I will use it sparingly.

Looks like I might have a chance to work on this some more this weekend. Wife and kiddo are taking off to Gran's house. I have been studying that application note thoroughly. Need to go back through the ELECHOUSE driver repo and figure out which GDO is clock when used with sync mode.

Nice thing with reading directly from the pins is I won't have to worry about the FIFO buffer overflow/underflow.

I already bought a whole load of adafruit QT boards for some miniaturization down the line with a new pcb. Will have to rework the code if (when) I get it working for the single core. One step at a time though.

@Crsarmv7l
Copy link
Contributor

Crsarmv7l commented Mar 23, 2024

Been a bit since I posted here and whew.

ELECHOUSE was a bust. It doesn't put out the clock, at least that I could find. I have been looking into radiolib again as @jpitz31 was, and it will do what I want it to do (bypass the FIFO entirely), it puts out the clock on gdo0 and data on gdo2 for sync serial but it has some significant differences with ELECHOUSE with regards to the registers.

So now I am at a crossroads. I can try and get the registers correct in ELECHOUSE to output the clock in sync which I can use to trigger reads of gdo2 data at appropriate intervals. The downside is zero support or updates.

or

I can shift to radiolib which has sync serial, and outputs clock to trigger pin reading, it is regularly updated and JGromes is great about answering questions. HOWEVER, I will have to rework my entire serial command line, as radiolib groups carriersense register settings in with sync word setup among others, and FEC cant be modified (I might be able to with God mode, but that has the potential to break other things)

I need to setup and try a simple program with radiolib before I go further.

On a different note I saw this #155

That was one of the first things I asked Flipper devs for, which they denied and the current cli usage for flipper is not very useful.

TBH I mostly use my Flipper as a tester for decoders/encoders I wrote for the Yardstick One 🤣 (Flipper code is ref upon ref upon ref, at least to me its very convoluted)

Some issues they will run into: Flipper uses async serial which avoids the fifo buffer which is good, but there is no clock associated so they will have to figure out a way to modulate the pin reads (IIRC async is 8x the bitrate? Although if that is true they could decimate the bits after rx). Anyway that would be a cool feature for the flipper

@Crsarmv7l
Copy link
Contributor

Crsarmv7l commented Mar 23, 2024

Hmm might be doable to modify ELECHOUSE

Radiolib registers for clock/data
image

Setup like this:
image

Elechouse registers
image

Looks like Elechouse just jumps immediately to async serial and has data on both GDO0 and GDO2 (the else statement, CCMode(1) is setup for the FIFO). Famous last words, but it might be as simple as adding another case to the CCMode and just setting IOCFG0 and IOCFG2 like radiolib does and omit MDMCFG3 & 4 changes (don't want to mess with the drate/chanbw since sync serial and drate/chanbw is entered as part of the cli entries)

I want to leave the base functionality of the lib intact, because I may need to use the FIFO's to tx once I get to that part.

@Crsarmv7l
Copy link
Contributor

Crsarmv7l commented Mar 24, 2024

Yep ELECHOUSE is out.

Radiolib does exactly what I want, at least for RX. Already forked and made a few changes to setPromiscuousMode to bring it more inline with lowball (1) with pulls pending for them.

Some of the things just make sense: for instance radiolib's promiscuous mode disables crc, syncword and preamble but doesn't set PQT=0. Upstream also currently doesn't allow you to set syncmode none/ syncmode carrier. So I changed it inline with radiolib's format to allow both of those and the pulls are pending.

Probably more changes required elsewhere, but I can actually use sync serial, and I will get to those as well.

Exciting Progress!

@Crsarmv7l
Copy link
Contributor

Upstream changes accepted. setPromiscuous now behaves like either lowball(0) or lowball(1).

Found a few more things I need to modify that might make it upstream. Then I can start integrating radiolib into the framework I was using with ELECHOUSE.

@Crsarmv7l
Copy link
Contributor

Crsarmv7l commented Mar 28, 2024

@atlas0fd00m ported your CalculatePktChanBw to Radiolib, ensured you were directly cited (maintainer wouldn't let me keep the OG name). Hope that's ok.

This, and my changes/merges with Promiscuous are part of getting my ducks in a row to use radiolib with my framework.

image

@Crsarmv7l
Copy link
Contributor

Major progress. A few more tests/fixes, gotta test the new pcbs coming, design a case, maybe do a pyserial frontend, and/or bleserial and I'll release everything to the world with @atlas0fd00m blessing. Shot you an email with a function test video and a way to chat directly.

@Crsarmv7l
Copy link
Contributor

Crsarmv7l commented Jun 2, 2024

adf5cb68-7149-46b6-93ba-9cea9e9e988a
a3e747c6-6c5e-4879-9ed7-17b6f8b65cdf
738201bd-6a59-4bcb-925b-3808af99a182

We are getting there boys. Final case, final pcb, and final hardware complete.

Basic functionality including keystop(), complete (last piece is an RSSI filter eg ignore signals below threshold).

Upstreamed multiple things to radiolib. Forked radiolib to adjust the buffer methodsfor this application.

Also need to write protocol decoders for a variety of signals (performed on chip).

Yardstick One functionality on any device (phone, tablet, whatever) with no drivers, YS1 commands/syntax, scriptable on a computer (with pyserial). Decoders (sometime in the next few months) built in. Downsides: tx/rx not as clean as a YS1 and no amp. Cant directly mess with the cc1101 registers with whatever you are using it on. I have spent far more making it, but total cost to someone for parts/case/PCB is about $25 using AliExpress and jlcpcb for the pcb and case printing.

@Crsarmv7l
Copy link
Contributor

Crsarmv7l commented Jun 10, 2024

Christ, I have been working on this for about 6 months so far.

Been on a bit of a tear recently:

Completed:

  • RSSI filter
    Notes: Since it is an SPI transaction I have to be careful with the usage so once it sees a spike above the desired RSSI threshold it fills the whole buffer and returns it. Since I am using Sync Serial and not the FIFO, Lowball 0 and 1 are working well. The RSSI filter also works well but you do get some noise at the end of the packet, as I can't efficiently check RSSI the whole time the packet is being rxed (maybe with a dual core ESP32 I could, but this is fine and very usable). Plus that would add a huge amount of extra complexity, and this is complex enough.

  • Hex string to byte conversion built in
    Notes: More of a pain than it should have been, but now entering a string like Rfxmit(ABCDEF) will send bytes 0xAB, 0xCD, 0xEF. Users will have to add 0's for single hex bytes, eg "0A" to send 0x0A. Radiolib does have some stuff built in but it wasn't working correctly as Serial input is ASCII, so it was sending some ASCII hex bytes. Not a knock on Radiolib, simply an issue with my method of input. More in the next:

  • RX displays full byte values
    Notes: Part of my problem above was an rx would show for instance: ABCDE. Is that 0xAB, 0xCD, 0x0D? Is it 0x0A, 0xBC, 0xDE? Is it 0x0A, 0x0B, 0x0C, 0x0D, 0x0E. You see where I am going with this. Too much ambiguity. When rxing two hex values will always be displayed eg 0A BC CD 0D. No more ambiguity.

Current problems / In Progress:

Radiolib doesn't have a very well flushed out Sync TX, and the FIFO only holds 63 bytes. HOWEVER, Ti DN500 discusses refilling the FIFO buffer as it transmits to send upto 255 bytes! A massive improvement:
https://www.ti.com/lit/an/swra109c/swra109c.pdf

I have been playing with refilling the buffer while txing, but due to my byte parsing issues above I couldn't verify it was performing as expected. Now that it is fixed, FIFO refill will be the next order of business.

Much later plans: Decoders. Gotta get basic functionality down first.

@Crsarmv7l
Copy link
Contributor

Crsarmv7l commented Jun 12, 2024

Partial Success using the FIFO to send up to 255bytes.

I developed the simplest possible method and test, and attempted to send 100 bytes of data using only the FIFO. I Rxed with an rtlsdr on a different computer and received 97 bytes of data.

The issue is I am using SPI to hammer the TXBYTES register and refill as space is available. SPI is slow and wasn't able to keep up as evidenced by the missing 3 bytes (FIFO ran dry). The TXBYTEs Register has some reliability issues as (discussed in the TI DN500). These tests were all done at a very low bitrate as well. A mere 2 Kbaud.

So this is not a viable option, but it does let me know this is possible.

The path forward on this is via interrupts. Ti is using the FIFO_THR as the trigger on IOCFG2 (GDO2) to trigger a FIFO refill. I need to play with it and see how it all works.

@Crsarmv7l
Copy link
Contributor

@atlas0fd00m You around?

I know the CC1111 doesn't have FIFO's so I was wondering how you are using the YS1 to send transmissions? Synchronous Serial? Async Serial? and if you could point me to the code where you do so.

I have made some progress using CC1101 FIFO's and still have more to test, but I am also looking for other alternatives.

@Crsarmv7l
Copy link
Contributor

Crsarmv7l commented Jun 14, 2024

Belay my last. Think I found what I was looking for. DMA with interrupts to trigger the feed...whew, can't say I am surprised but hoping I can find a way to not use that.

@Crsarmv7l
Copy link
Contributor

ISR's on my program working. Lets see how many bytes I can consistently get out the door with burst SPI transactions. Obviously the FIFO is pushing out faster than I can push in, but if a decent amount can go out.... I might just avoid trying DMA altogether.

This project can't and isn't supposed to replace well designed hardware and well written software like the YS1 and Rfcat. Just be a cheap and decently functional device that anyone who knows rfcat can pick up and get working.

@Crsarmv7l
Copy link
Contributor

Crsarmv7l commented Jun 18, 2024

Been a rollercoaster with refilling FIFO. Was working terribly even with interrupts for FIFO_THR below threshold and FIFO full. Got a bit of a sanity check from the dev behind radiolib and instead wrote just a very simple sketch that only tx's and refills the FIFO. I used 100bytes of 0xAA to test.

It worked great! 99.5 bytes out of 100 Rxed on the first signal and 98 bytes out of 100 Rxed on the second signal. So it is something with my multiple loops that is killing the throughput (I was getting between 85-93 bytes throughput without much consistency).

Got a testing setup done last night in my program, bypassing Radiolib calls and doing the SPI transactions to fill the FIFO and run the interrupts directly. Lets see how that works tonight/tomrrow, then maybe I can wrap it in a func.

In theory it should work... key word "in theory". This is the last remaining piece for basic functionality (basic functionality is fine up to 64 bytes, I am shooting for at least 100 bytes tx capability. Probably not needed as 64 bytes=512 bits already. But for some reason I want it.)

I should probably also create a func to send bitstrings.

@Crsarmv7l
Copy link
Contributor

Crsarmv7l commented Jun 24, 2024

Created a method for sending bitstrings....These FIFO refills are killing me though. Sometimes it works decently, sometimes it doesn't.

Drafting up some real test bitstrings... yeah I need the refills. Sigh this is probably gonna be really hard

@Crsarmv7l
Copy link
Contributor

Getting much more consistent with byte throughput. New problem though. The signal sometimes gets inverted. I have no idea how that is happening as I am filling the FIFO and the FIFO should be emptying in accordance with the cc1101 clock...

@Crsarmv7l
Copy link
Contributor

Crsarmv7l commented Jul 14, 2024

URH of the issue. 13 signals 12 of which are perfect in length and content. 100bytes in total, so a FIFO refill is done. One signal misses some clocks and inverts.

This is at 8 kilobaud, so very usable now

Screenshot 2024-07-11 7 03 19 PM

I have run out of things to test, but it is a lot more reliable than it used to be. Previously I was getting issues on 1 signal in 4. Enabling the freq synth and giving it some time prior to tx drastically reduces the error rate for some reason.

1 in 13 isn't terrible, and I may just include 4-5 repeats to ensure a majority of signals are correct and accurate.

@Crsarmv7l
Copy link
Contributor

After much testing it seems to be voltage droop. ESP32-S2 is powered from the usb port and the cc1101 is powered from the esp32-s2 3v3 rail. Freq synth + calibration helps to somewhat compensate fro the droop. Looking into power smoothing and caps.

Def not usb noise as I tested with shielded usb with ferrites.

@Crsarmv7l
Copy link
Contributor

A small ceramic cap 0.1uf seems to be helping quite a lot. Still have more to dial in though. I am running the ESP32-S2 at 160Mhz (less power) but want to push it up to the full 240Mhz. Likewise I am currently running SPI at 4Mhz, but want to push it up to the datasheet max of 6.5Mhz (burst).

Probably will try other caps like 0.15uf and 0.22uf as well.

@Crsarmv7l
Copy link
Contributor

Still at work on this. Trying to completely eliminate the hw issue and going through a move so I am taking a slight break.

@Crsarmv7l
Copy link
Contributor

Still working. First functional test with a known signal receiver from serial interface.

Shows radio setup works, commands are recognized, signal parsing/tx is accurate... All from turning a light on/off. Had to trim the video to fit so only off is shown.

TRIM_20240910_162844.mp4

Currently working on the python frontend to allow scriptability.

I have had to make a couple small deviations from rfcat for sanity and still working on the random issue with a small percentage of tx's greater than 64 bytes.

@Crsarmv7l
Copy link
Contributor

Crsarmv7l commented Sep 14, 2024

The above was with the serial interface using a serial monitor.

Last night I made enough progress with the python frontend to take an RfCat script I had written for the Princeton Protocol and only removing the bitstring library (used to create bytes for RfCat) successfully transmit a princeton signal as desired with no other changes.

That is one of the differences, my tool assumes all strings are either hex or binary (checks, padding, and conversion to bytes are done in my back end).

Multiple inits and destructions ( d=RfCat()) are handled gracefully and each scripted command is evaluated in the python frontend lib to ensure there are no radio issues (radio/ESP32 returns are verified)

I have 3 more PCB versions in the process of being created and shipped. To hopefully eliminate what I believe is a hardware issue that causes a missed clock occasionally when sending signals greater than 64 bytes. To combat this I have done the following:
- Ground Planes to eliminate potential cross talk and EMI
- Wider Power trace
- Better trace routing

Each PCB version of the 3 has slight differences that will be helpful with testing.

Remaining todo:
- RX piece for python lib (already working in Serial Monitor). Will need to use multithreading in order to accommodate keystop() which is already done in the backend.
- My RX will likely function differently to RfCat as receive until keystop() is my default RX setting.
- Additional differences setModeRX/IDLE/TX are not used I always go back to IDLE and perform calibration upon change to RX or TX
- In practice with an existing RfCat script, this will simply pass, but a newly created script won't need those commands.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

No branches or pull requests

3 participants