Compare commits
23 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
af1fa134cc | ||
|
|
812a55a381 | ||
|
|
41ddf0247a | ||
|
|
f3c3a214e3 | ||
|
|
5536368ba7 | ||
|
|
e2063420e9 | ||
|
|
cfaf24357c | ||
|
|
543e7d39e0 | ||
|
|
f9981b1cec | ||
|
|
d36d0132a8 | ||
|
|
b29775dfbe | ||
|
|
8a8dfcb190 | ||
|
|
25600e48fb | ||
|
|
7c06c99bd1 | ||
|
|
ad3142da95 | ||
|
|
0c14b6d2b9 | ||
|
|
4522b9261e | ||
|
|
144c9bbf5a | ||
|
|
20cf246eda | ||
|
|
02e0d4837e | ||
|
|
816720e15c | ||
|
|
c60e0334bd | ||
|
|
ade0cdc3cd |
@@ -11,7 +11,7 @@ let test3Strings: String = testString + "\n" + testString + "\n" + testString +
|
|||||||
|
|
||||||
let arguments = CommandLine.arguments
|
let arguments = CommandLine.arguments
|
||||||
guard arguments.count >= 2 else {
|
guard arguments.count >= 2 else {
|
||||||
print("Need serial port name, e.g. /dev/ttyUSB0 as the first argument.")
|
print("Need serial port name, e.g. /dev/ttyUSB0 or /dev/cu.usbserial as the first argument.")
|
||||||
exit(1)
|
exit(1)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
8
Examples/SwiftSerialIM/Package.swift
Normal file
8
Examples/SwiftSerialIM/Package.swift
Normal file
@@ -0,0 +1,8 @@
|
|||||||
|
import PackageDescription
|
||||||
|
|
||||||
|
let package = Package(
|
||||||
|
name: "SwiftSerialIM",
|
||||||
|
dependencies: [
|
||||||
|
.Package(url: "https://github.com/yeokm1/SwiftSerial.git", majorVersion: 0)
|
||||||
|
]
|
||||||
|
)
|
||||||
128
Examples/SwiftSerialIM/Sources/main.swift
Normal file
128
Examples/SwiftSerialIM/Sources/main.swift
Normal file
@@ -0,0 +1,128 @@
|
|||||||
|
import Foundation
|
||||||
|
import SwiftSerial
|
||||||
|
|
||||||
|
|
||||||
|
let arguments = CommandLine.arguments
|
||||||
|
guard arguments.count >= 2 else {
|
||||||
|
print("Need serial port name, e.g. /dev/ttyUSB0 or /dev/cu.usbserial as the first argument.")
|
||||||
|
exit(1)
|
||||||
|
}
|
||||||
|
|
||||||
|
print("Connect a null modem serial cable between two machines before you continue to use this program")
|
||||||
|
|
||||||
|
let portName = arguments[1]
|
||||||
|
let serialPort: SerialPort = SerialPort(path: portName)
|
||||||
|
|
||||||
|
var myturn = true
|
||||||
|
|
||||||
|
// Prepares the stdin so we can getchar() without echoing
|
||||||
|
func prepareStdin() {
|
||||||
|
|
||||||
|
// Set up the control structure
|
||||||
|
var settings = termios()
|
||||||
|
|
||||||
|
// Get options structure for stdin
|
||||||
|
tcgetattr(STDIN_FILENO, &settings)
|
||||||
|
|
||||||
|
//Turn off ICANON and ECHO
|
||||||
|
settings.c_lflag &= ~tcflag_t(ICANON | ECHO)
|
||||||
|
|
||||||
|
tcsetattr(STDIN_FILENO, TCSANOW, &settings)
|
||||||
|
}
|
||||||
|
|
||||||
|
func getKeyPress () -> UnicodeScalar {
|
||||||
|
let valueRead: Int = Int(getchar())
|
||||||
|
|
||||||
|
guard let charRead = UnicodeScalar(valueRead) else{
|
||||||
|
return UnicodeScalar("")!
|
||||||
|
}
|
||||||
|
|
||||||
|
return charRead
|
||||||
|
}
|
||||||
|
|
||||||
|
func printToScreenFrom(myself: Bool, characterToPrint: UnicodeScalar){
|
||||||
|
|
||||||
|
if(myturn && !myself){
|
||||||
|
myturn = false
|
||||||
|
print("\n\nOther: ", terminator:"")
|
||||||
|
} else if (!myturn && myself){
|
||||||
|
myturn = true
|
||||||
|
print("\n\nMe: ", terminator:"")
|
||||||
|
}
|
||||||
|
|
||||||
|
print(characterToPrint, terminator:"")
|
||||||
|
}
|
||||||
|
|
||||||
|
func backgroundRead() {
|
||||||
|
while true{
|
||||||
|
do{
|
||||||
|
let readCharacter = try serialPort.readChar()
|
||||||
|
printToScreenFrom(myself: false, characterToPrint: readCharacter)
|
||||||
|
} catch {
|
||||||
|
print("Error: \(error)")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
do {
|
||||||
|
|
||||||
|
print("Attempting to open port: \(portName)")
|
||||||
|
try serialPort.openPort()
|
||||||
|
print("Serial port \(portName) opened successfully.")
|
||||||
|
defer {
|
||||||
|
serialPort.closePort()
|
||||||
|
print("Port Closed")
|
||||||
|
}
|
||||||
|
|
||||||
|
serialPort.setSettings(receiveRate: .baud9600,
|
||||||
|
transmitRate: .baud9600,
|
||||||
|
minimumBytesToRead: 1)
|
||||||
|
|
||||||
|
prepareStdin()
|
||||||
|
|
||||||
|
|
||||||
|
//Turn off output buffering if not multiple threads will have problems printing
|
||||||
|
setbuf(stdout, nil);
|
||||||
|
|
||||||
|
|
||||||
|
//Run the serial port reading function in another thread
|
||||||
|
#if os(Linux)
|
||||||
|
var readingThread = pthread_t()
|
||||||
|
|
||||||
|
let pthreadFunc: @convention(c) (UnsafeMutableRawPointer?) -> UnsafeMutableRawPointer? = {
|
||||||
|
observer in
|
||||||
|
|
||||||
|
backgroundRead()
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
pthread_create(&readingThread, nil, pthreadFunc, nil)
|
||||||
|
|
||||||
|
#elseif os(OSX)
|
||||||
|
DispatchQueue.global(qos: .userInitiated).async {
|
||||||
|
backgroundRead()
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
||||||
|
|
||||||
|
print("\nReady to send and receive messages in realtime!")
|
||||||
|
print("\nMe: ", terminator:"")
|
||||||
|
|
||||||
|
|
||||||
|
while true {
|
||||||
|
var enteredKey = getKeyPress()
|
||||||
|
printToScreenFrom(myself: true, characterToPrint: enteredKey)
|
||||||
|
var _ = try serialPort.writeChar(enteredKey)
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
} catch PortError.failedToOpen {
|
||||||
|
print("Serial port \(portName) failed to open. You might need root permissions.")
|
||||||
|
} catch {
|
||||||
|
print("Error: \(error)")
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
51
README.md
51
README.md
@@ -9,13 +9,21 @@ This library is an improvement over my previous now deprecated library [SwiftLin
|
|||||||
<a href="https://developer.apple.com/swift"><img src="https://img.shields.io/badge/swift3-compatible-orange.svg?style=flat" alt="Swift 3 compatible" /></a>
|
<a href="https://developer.apple.com/swift"><img src="https://img.shields.io/badge/swift3-compatible-orange.svg?style=flat" alt="Swift 3 compatible" /></a>
|
||||||
<a href="https://raw.githubusercontent.com/uraimo/SwiftyGPIO/master/LICENSE"><img src="http://img.shields.io/badge/license-MIT-blue.svg?style=flat" alt="License: MIT" /></a>
|
<a href="https://raw.githubusercontent.com/uraimo/SwiftyGPIO/master/LICENSE"><img src="http://img.shields.io/badge/license-MIT-blue.svg?style=flat" alt="License: MIT" /></a>
|
||||||
|
|
||||||
|
## Talk on this library
|
||||||
|
|
||||||
|
I gave a talk on this library and one of its examples SwiftSerialIM. Click on the links below to see the slides and video.
|
||||||
|
|
||||||
|
[](http://www.slideshare.net/yeokm1/a-science-project-swift-serial-chat)
|
||||||
|
|
||||||
|
[](https://www.youtube.com/watch?v=6PWP1eZo53s)
|
||||||
|
|
||||||
## Mac OS Preparation
|
## Mac OS Preparation
|
||||||
|
|
||||||
You should have Xcode 8 installed with the command line tools.
|
You should have Xcode 8 installed with the command line tools.
|
||||||
|
|
||||||
## Linux System Preparation
|
## Linux System Preparation
|
||||||
|
|
||||||
Before using this library, I assume you already have Ubuntu installed and fully updated on your system or single-board computer. To get Ubuntu installed on the Raspberry Pi, use this [link](https://wiki.ubuntu.com/ARM/RaspberryPi).
|
Before using this library, I assume you already have Ubuntu installed and fully updated on your system or single-board computer. To get Ubuntu installed on the Raspberry Pi, use this [link](https://wiki.ubuntu.com/ARM/RaspberryPi).
|
||||||
|
|
||||||
### Install Swift 3 on Ubuntu on x86-based machines
|
### Install Swift 3 on Ubuntu on x86-based machines
|
||||||
|
|
||||||
@@ -50,17 +58,20 @@ sudo update-alternatives --install /usr/bin/clang++ clang++ /usr/bin/clang++-3.6
|
|||||||
|
|
||||||
cd ~
|
cd ~
|
||||||
#Replace the link below with the latest version
|
#Replace the link below with the latest version
|
||||||
wget http://swift-arm.ddns.net/job/Swift-3.0-Pi3-ARM-Incremental/lastSuccessfulBuild/artifact/swift-3.0-2016-10-13-RPi23-ubuntu16.04.tar.gz
|
wget http://swift-arm.ddns.net/job/Swift-3.0-Pi3-ARM-Incremental/lastSuccessfulBuild/artifact/swift-3.0-2016-10-13-RPi23-ubuntu16.04.tar.gz
|
||||||
mkdir swift-3.0
|
mkdir swift-3.0
|
||||||
cd swift-3.0 && tar -xzf ../swift-3.0-2016-10-13-RPi23-ubuntu16.04.tar.gz
|
cd swift-3.0 && tar -xzf ../swift-3.0-2016-10-13-RPi23-ubuntu16.04.tar.gz
|
||||||
|
|
||||||
#This command can be added to your bash profile so Swift will be in your PATH after a reboot
|
#This command can be added to your bash profile so Swift will be in your PATH after a reboot
|
||||||
nano ~/.profile
|
nano ~/.profile
|
||||||
export PATH=$HOME/swift-3.0/usr/bin:$PATH
|
export PATH=$HOME/swift-3.0/usr/bin:$PATH
|
||||||
```
|
```
|
||||||
## Jumping straight into sample code
|
## Jumping straight into sample code
|
||||||
|
To get started quickly, you can take a look at my example projects [here](Examples/).
|
||||||
|
|
||||||
To get started quickly, you can take a look at my example project [here](Examples/SwiftSerialExample). In order to run the example properly, you need to connect one of your (USB/UART) serial ports in a loopback manner. Basically, you short the TX and RX pins of the serial port.
|
### Example 1: Loopback Test
|
||||||
|
|
||||||
|
In order to run this example properly, you need to connect one of your (USB/UART) serial ports in a loopback manner. Basically, you short the TX and RX pins of the serial port. This library currently only support the `/dev/cu.*` variant on Mac. Read the beginning of the API usage section for more details.
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
git clone https://github.com/yeokm1/SwiftSerial.git
|
git clone https://github.com/yeokm1/SwiftSerial.git
|
||||||
@@ -71,11 +82,28 @@ swift build
|
|||||||
sudo ./.build/debug/SwiftSerialExample /dev/ttyUSB0
|
sudo ./.build/debug/SwiftSerialExample /dev/ttyUSB0
|
||||||
|
|
||||||
#For Mac: Root is not required
|
#For Mac: Root is not required
|
||||||
./.build/debug/SwiftSerialExample /dev/tty.usbserial
|
./.build/debug/SwiftSerialExample /dev/cu.usbserial
|
||||||
|
|
||||||
#If all goes well you should see a series of messages informing you that data transmitted has been received properly.
|
#If all goes well you should see a series of messages informing you that data transmitted has been received properly.
|
||||||
```
|
```
|
||||||
|
|
||||||
|
### Example 2: A chat app between 2 machines
|
||||||
|
|
||||||
|
In order to run this example properly, you need 2 machines connected by a [null-modem cable](https://en.wikipedia.org/wiki/Null_modem) or 2 USB-Serial adapters with the TX-RX pins connected to each other. Run a copy of my program on both machines.
|
||||||
|
|
||||||
|
```bash
|
||||||
|
git clone https://github.com/yeokm1/SwiftSerial.git
|
||||||
|
cd SwiftSerial/Examples/SwiftSerialIM/
|
||||||
|
swift build
|
||||||
|
|
||||||
|
#For Linux: You need root to access the serial port. Replace /dev/ttyUSB0 with the name of your serial port under test
|
||||||
|
sudo ./.build/debug/SwiftSerialIM /dev/ttyUSB0
|
||||||
|
|
||||||
|
#For Mac: Root is not required
|
||||||
|
./.build/debug/SwiftSerialIM /dev/cu.usbserial
|
||||||
|
```
|
||||||
|
People at both machines can now "chat" with each other.
|
||||||
|
|
||||||
## Integrating with your project
|
## Integrating with your project
|
||||||
|
|
||||||
Add SwiftSerial as a dependency to your project by editing the `Package.swift` file.
|
Add SwiftSerial as a dependency to your project by editing the `Package.swift` file.
|
||||||
@@ -102,7 +130,9 @@ Then run `swift build` to download the dependencies and compile your project. Yo
|
|||||||
```swift
|
```swift
|
||||||
let serialPort: SerialPort = SerialPort(path: portName)
|
let serialPort: SerialPort = SerialPort(path: portName)
|
||||||
```
|
```
|
||||||
Supply the portname that you wish to open like `/dev/ttyUSB0` or `/dev/tty.usbserial`.
|
Supply the portname that you wish to open like `/dev/ttyUSB0` or `/dev/cu.usbserial`.
|
||||||
|
|
||||||
|
For Macs, this library currently only works with the `/dev/cu.*` ports instead of the `/dev/tty.*`. I have enabled blocking on the serial port to prevent high CPU usage which will prevent the `/dev/tty.*` from working. Read more about the differences between the two [here](http://stackoverflow.com/questions/8632586/macos-whats-the-difference-between-dev-tty-and-dev-cu). If there is a problem, open an issue describing your situation and let me look into it.
|
||||||
|
|
||||||
### Opening the Serial Port
|
### Opening the Serial Port
|
||||||
|
|
||||||
@@ -117,7 +147,7 @@ Opening the port without any parameters will set the port to receive and transmi
|
|||||||
```swift
|
```swift
|
||||||
serialPort.setSettings(receiveRate: .baud9600, transmitRate: .baud9600, minimumBytesToRead: 1)
|
serialPort.setSettings(receiveRate: .baud9600, transmitRate: .baud9600, minimumBytesToRead: 1)
|
||||||
```
|
```
|
||||||
The port settings call can be as simple as the above. For the baud rate, just supply both transmit and receive even if you are only intending to use one transfer direction. For example, transmitRate will be ignored if you specified `andTransmit : false` when opening the port.
|
The port settings call can be as simple as the above. For the baud rate, just supply both transmit and receive even if you are only intending to use one transfer direction. For example, transmitRate will be ignored if you specified `andTransmit : false` when opening the port.
|
||||||
|
|
||||||
`minimumBytesToRead` determines how many characters the system must wait to receive before it will return from a [read()](https://linux.die.net/man/2/read) function. If in doubt, just put 1.
|
`minimumBytesToRead` determines how many characters the system must wait to receive before it will return from a [read()](https://linux.die.net/man/2/read) function. If in doubt, just put 1.
|
||||||
|
|
||||||
@@ -166,10 +196,15 @@ func readUntilChar(_ terminator: CChar) throws -> String
|
|||||||
```
|
```
|
||||||
Keep reading until the specified CChar is encountered. Return the string read so far without that value.
|
Keep reading until the specified CChar is encountered. Return the string read so far without that value.
|
||||||
|
|
||||||
|
```swift
|
||||||
|
func readByte() throws -> UInt8
|
||||||
|
```
|
||||||
|
Read only one byte. This works best if `minimumBytesToRead` has been set to `1` when opening the port. This function internally calls `readBytes()`.
|
||||||
|
|
||||||
```swift
|
```swift
|
||||||
func readChar() throws -> UnicodeScalar
|
func readChar() throws -> UnicodeScalar
|
||||||
```
|
```
|
||||||
Read only one character. This works best if `minimumBytesToRead` has been set to `1` when opening the port. This function internally calls `readBytes()`.
|
Read only one character. This works best if `minimumBytesToRead` has been set to `1` when opening the port. This function internally calls `readByte()`.
|
||||||
|
|
||||||
### Writing data to the port
|
### Writing data to the port
|
||||||
|
|
||||||
|
|||||||
File diff suppressed because it is too large
Load Diff
BIN
first-slide.png
Normal file
BIN
first-slide.png
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 311 KiB |
BIN
swift-serial-talk-slides.pptx
Normal file
BIN
swift-serial-talk-slides.pptx
Normal file
Binary file not shown.
Reference in New Issue
Block a user