Upgrade to Pro
— share decks privately, control downloads, hide ads and more …
Speaker Deck
Features
Speaker Deck
PRO
Sign in
Sign up for free
Search
Search
TinyGoでIoTを始めよう
Search
irieda
April 20, 2019
Technology
2
850
TinyGoでIoTを始めよう
TinyGoというGo言語サブセットを使って
PC-OSの無い環境の組み込み開発を始める方法の紹介。
irieda
April 20, 2019
Tweet
Share
More Decks by irieda
See All by irieda
SpaGoでSPAを作ろう!
nobonobo
0
610
Goだけでロビーサーバーを作ってみた
nobonobo
2
540
GopherJS+Vecty
nobonobo
1
940
Other Decks in Technology
See All in Technology
Liquid Glass革新とSwiftUI/UIKit進化
fumiyasac0921
0
120
実践! AIエージェント導入記
1mono2prod
0
140
AIエージェント最前線! Amazon Bedrock、Amazon Q、そしてMCPを使いこなそう
minorun365
PRO
10
3.8k
25分で解説する「最小権限の原則」を実現するための AWS「ポリシー」大全 / 20250625-aws-summit-aws-policy
opelab
6
690
第9回情シス転職ミートアップ_テックタッチ株式会社
forester3003
0
130
Amazon ECS & AWS Fargate 運用アーキテクチャ2025 / Amazon ECS and AWS Fargate Ops Architecture 2025
iselegant
13
4.3k
AIの最新技術&テーマをつまんで紹介&フリートークするシリーズ #1 量子機械学習の入門
tkhresk
0
120
キャディでのApache Iceberg, Trino採用事例 -Apache Iceberg and Trino Usecase in CADDi--
caddi_eng
0
170
米国国防総省のDevSecOpsライフサイクルをAWSのセキュリティサービスとOSSで実現
syoshie
2
790
LinkX_GitHubを基点にした_AI時代のプロジェクトマネジメント.pdf
iotcomjpadmin
0
160
ObsidianをMCP連携させてみる
ttnyt8701
2
140
エンジニア向け技術スタック情報
kauche
0
110
Featured
See All Featured
Building Better People: How to give real-time feedback that sticks.
wjessup
367
19k
[Rails World 2023 - Day 1 Closing Keynote] - The Magic of Rails
eileencodes
35
2.3k
Learning to Love Humans: Emotional Interface Design
aarron
273
40k
Keith and Marios Guide to Fast Websites
keithpitt
411
22k
Rails Girls Zürich Keynote
gr2m
94
14k
Intergalactic Javascript Robots from Outer Space
tanoku
271
27k
Building Adaptive Systems
keathley
43
2.6k
The Invisible Side of Design
smashingmag
299
51k
Dealing with People You Can't Stand - Big Design 2015
cassininazir
367
26k
Templates, Plugins, & Blocks: Oh My! Creating the theme that thinks of everything
marktimemedia
31
2.4k
4 Signs Your Business is Dying
shpigford
184
22k
The Cult of Friendly URLs
andyhume
79
6.4k
Transcript
TinyGoͰIoTΛ࢝ΊΑ͏
͓લ୭Αʁ ϝΧτϩιϑτ Pythonista -> Gopher झຯͰGoʹ͔ͳ͍ͷ։ Goྺ̒ऑ͘Β͍ αΠτ: ձࣾ: 144Lab(2017/07/01
͔Β৽໊ࣾ) HN: @nobonobo http://golang.rdy.jp/
ΈࠐΈ։ൃͬͯʁ
ΈࠐΈͱݴͬͯʁ େผͯ͠ PC͚OSͷ্ͰΈࠐΈ PC͚OSແ͠ͰΈࠐΈ ʢΈࠐΈ͚OS·ͨOSແ͠ʣ ͱ͍͏̎छྨ͕͋Γ·͢ɻ
PC͚OSͷ্ͰΈࠐΈ PCʹ͍ۙখܕϘʔυʢRaspberryPiͳͲʣͰ LinuxWindows10ͷ্ʹ ΞϓϦΛ࣮͢ΔελΠϧ ͕ૉΒ͍͠ ͨͩ͠ɺిݯ֬อ͕ඞਢ Ϟϊͷબࢶ࣮࣭̎ RaspberryPiܥ ߴՁͳϘʔυͷྨ https://gobot.io
PC͚OSແ͠ͰΈࠐΈ Goͷෆಘҙ Goͷཪํͷػೳηοτ͕OSʹґଘ goroutineͷεέδϡʔϥGCͳͲ OSͷແ͍ڥʹҠ২͢Δίετ͕ߴ͍ ϨΠϠͷI/Oࢧԉػೳ͕ͱͱແ͍ PCͱҧ͍ཧతʹଟ༷͗͢Δڥ
ΈࠐΈ։ൃͷੈք ιϑτΣΞͷ͕ࣝٻΊΒΕΔ ϨΠϠʔͷࣝٻΊΒΕΔ ϋʔυΣΞͷࣝগ͠ٻΊΒΕΔ ։ൃڥΛ͑Δͷʹίετ͕͔͔Δ ΈࠐΈ is Hard!
ͦͯ͠ɾɾɾ C/C++͕σϑΝΫτελϯμʔυ C͚ͩͳΒ·ͩ͠C++πϥΠ ॲཧͷ࣮ମΛӅ͢ͷ্͕ख͗͢Δ ࠔͳґଘղܾ includeࠈͱIFDEFࠈ ΦʔόʔϩʔυʹςϯϓϨʔτ ΈࠐΈ is Hard!
C/C++ͷΘΓʹGoͳΒʁ ϦʔμϏϦςΟ࠷ߴʂ ΤϥʔΛపఈͯ͠νΣοΫ͢ΔจԽʂ undefined-behavierʹ·͞Εͳ͍ʂ ґଘղܾʹgo-get͕͑Δʂ ΈࠐΈ͕গۙ͠ʹͳΔ͔ʁʂ
ͱ͍͏Θ͚Ͱ TinyGoͰ͢Αʂʂ
ઌ݄ϗοςϯτϦೖΓ https://tech.144lab.com/entry/tinygo
λʔήοτྫ
micro:bit ରԠ։ൃπʔϧͷ͕τοϓΫϥε ϒϩοΫํࣜͰখֶੜϓϩάϥϛϯάOK ϒϩοΫʹ͖ͨΒJS,TS ArduinoͰPythonͰRuby,Jacvascript,LuaͳͲͳͲ ϓϩ͚SEGGER Embedded StudioͰOK
circuit playground express ࡌΪϛοΫ͕τοϓΫϥε ϑϧΧϥʔ֊ௐ͖ͭLED̍̌ݸ εϐʔΧʔʹϚΠΫ ޫྔηϯαɺԹɾ࣪ηϯα ̒࣠ʢՃɾ֯ʣηϯα IRϦϞίϯૹड৴ λονηϯγϯά
UART/USB/SPI/I2C/etc...
࣮ࡍʹ͍ΖΜͳίʔυΛ ίϯύΠϧͯ͜͠ΕΒͰ ಈ͔ͯ͠Έ·͠ΐ͏ʂ
TinyGoͷ͍ํ dockerϕʔε͕ࠓΦεεϝͰ͢ɻ ʢdevϒϥϯνϏϧυେม ˍ ߴසͰґଘมߋ͞Ε Δʣ $ docker run -it
--rm tinygo/tinygo tinygo version tinygo version 0.4.1 linux/amd64
ͨ·ʹΞοϓσʔτͯ͠Έ·͠ΐ͏ $ docker pull tinygo/tinygo:latest latest: Pulling from tinygo/tinygo Digest:
sha256:821ad71e7d9b3fe46a84a5818e52d3d3596e0d8f3855f8407b917a37371b2b23 Status: Image is up to date for tinygo/tinygo:latest
sample̍ ͍ΘΏΔLνΧʢΈࠐΈͰͷHelloWorldʣ package main import ( "machine" "time" ) func
main() { led := machine.GPIO{machine.LED} led.Configure(machine.GPIOConfig{Mode: machine.GPIO_OUTPUT}) for { led.Low() time.Sleep(time.Millisecond * 500) led.High() time.Sleep(time.Millisecond * 500) } }
Ϗϧυ output.uf2 --Drag&Drop--> CIRCUITPYσόΠε ͜Ε͚ͩͰλʔήοτʹॻ͖ࠐ·ΕΔ $ go mod init github.com/nobonobo/examples/sample1
$ PACKAGE_PATH = github.com/nobonobo/examples/sample1 $ PROJECT_PATH = /go/src/$PACKAGE_PATH $ docker run -it --rm $PWD:$PROJECT_PATH -w $PROJECT_PATH \ tinygo/tinygo tinygo build -target=circuitplay-express -o output.uf2 .
TinyGoͷఏڙόε࣮(1) ڞ௨Ͱఏڙ͢Δόε GPIO(൚༻ೖग़ྗϐϯ) ADCʢΞφϩάܭଌʣ PWMʢαʔϘɺύϫʔίϯτϩʔϧ͚ʣ UARTʢPCͱͷ࿈ܞ༻ʣ SPIʢখ͞ͳσόΠεͱͷ࿈ܞ༻ʣ I2Cʢখ͞ͳσόΠεͱͷ࿈ܞ༻ʣ
TinyGoͷఏڙόε࣮(2) Ұ෦ͷλʔήοτ DACʢΞφϩάిѹൃੜʣ
ΞʔΩςΫνϟผͷఏڙػೳ ϨδελΞΫηε ΠϯϥΠϯΞηϯϒϥ λΠϚʔɺUARTϋϯυϦϯά ΄ͱΜͲͷλʔήοτͰ ʮprintlnʯͰ γϦΞϧϙʔτʹσόοάϩάΛग़ྗͰ͖Δɻ
पลσόΠεผυϥΠόʔ Device Name Interface Type SPI I2C I2C I2C GPIO
UART I2C I2C I2C GPIO https://github.com/tinygo-org/drivers APA102 RGB LED BH1750 ambient light sensor BlinkM RGB LED BMP180 barometer "Easystepper" stepper motor controller ESP8266/ESP32 AT Command set for WiFi/TCP/UDP MAG3110 magnetometer MMA8653 accelerometer MPU6050 accelerometer/gyroscope WS2812 RGB LED
֎෦ύοέʔδͱdocker ґଘ͢ΔύοέʔδΛՃ͍ͨ͠ʂʂ ͋ΕʁdockerϕʔεͩͱͲ͏͢Δʁʂ ͦ͜Ͱgo modͰ͢Α $ export GO111MODULE=on $ go
mod github.com/..../sample1 $ go get github.com/tinygo-org/drivers/ws2812 $ go mod vendor
͢Δͱʁ vendor/ github.com/ tinygo-org/ drivers/ ws2812/ main.go ͖ͬࣔͨ͞͠dockerίϚϯυͰίϯςφ͔Β ws2812ύοέʔδ͕ࢀরՄೳʹͳΔɻ
LEDετϦοϓυϥΠόΛ࡞Δ package main import ( "device/arm" "image/color" "machine" "github.com/tinygo-org/drivers/ws2812" )
// NeopixelDriver represents a connection to a NeoPixel type NeopixelDriver struct { pin machine.GPIO device ws2812.Device pixelCount int pixels []color.RGBA } // NewNeopixelDriver returns a new NeopixelDriver
ͦͯ͠ಈ͔ͯ͠ΈΔ package main import ( "image/color" "machine" "time" ) const
NUM_PIXELS = 10 func main() { neo := NewNeopixelDriver(machine.GPIO{machine.NEOPIXELS}, NUM_PIXELS) n := 0 var pows = []uint8{255, 127, 63, 31, 15, 7, 3, 1, 0, 0} for { for i := 0; i < NUM_PIXELS; i++ { pow := pows[NUM_PIXELS-1-i] neo.SetPixel((n+i)%NUM_PIXELS, color.RGBA{R: pow, G: pow, B: pow}) }
DEMO ͕ʔɾɾ͕ʔʂʂ
໌Δ͞ʹԠͯ͡LEDΛ౮͢ package main import ( "image/color" "machine" "time" ) const
NUM_PIXELS = 10 func main() { machine.InitADC() sensor := machine.ADC{machine.A8} sensor.Configure() neo := NewNeopixelDriver(machine.GPIO{machine.NEOPIXELS}, NUM_PIXELS) for { for i := 0; i < NUM_PIXELS; i++ { pow := uint8(sensor.Get() / 256) neo.SetPixel(i, color.RGBA{R: pow, G: pow, B: pow})
DEMO
GoͬΆ͘ॻ͘ package main import ( "image/color" "machine" "time" ) const
NUM_PIXELS = 10
Ϙλϯࢹؔ func observer(event chan int) { btnA := machine.GPIO{machine.BUTTONA} btnA.Configure(machine.GPIOConfig{Mode:
machine.GPIO_INPUT_PULLDOWN}) btnB := machine.GPIO{machine.BUTTONB} btnB.Configure(machine.GPIOConfig{Mode: machine.GPIO_INPUT_PULLDOWN}) next := make([]bool, 2) prev := make([]bool, 2) prev[0], prev[1] = btnA.Get(), btnB.Get() for { time.Sleep(20 * time.Millisecond) next[0], next[1] = btnA.Get(), btnB.Get() for i := range next { if !prev[i] && next[i] { event <- i } } prev[0], prev[1] = next[0], next[1] } }
ϨϯμϦϯάؔ func renderer(ch chan *NeopixelDriver) { for { v :=
<-ch v.Show() } }
mainؔ func main() { events := make(chan int) go observer(events)
// ϘλϯࢹλεΫىಈ ɹneo := NewNeopixelDriver(machine.GPIO{machine.NEOPIXELS}, NUM_PIXELS) ch := make(chan *NeopixelDriver) go renderer(ch) // ϨϯμϥʔλεΫىಈ prev, next := 0, 0 neo.SetPixel(prev, color.RGBA{R: 10, G: 10, B: 10}) ch <- neo // ϨϯμϦϯάΛൃՐ for { select { case v := <-events: switch v { case 0: println("buttonA pressed") next++ if next >= NUM_PIXELS {
DEMO
·ͱΊ ฒߦॲཧ͕ॻ͖͍͢ʂ ݱঢ়ͷϨΠϠ֦ுखஈ ΠϯϥΠΞηϯϒϥ SVCall CGOαϙʔτߴϨϕϧαϙʔτ͕ෆ TinyGoͷϏϧυͰgo-mod·ͩ͑ͳ͍ ͕ɺdockerͱͷซ༻ͰͳΜͱ͔ͳΔ
ࠓޙͷൃల
׆ൃͳ׆ಈ BLEϥΠϒϥϦରԠ CGOαϙʔτ ՃηϯαͳͲͷυϥΠόʔ ESPγϦʔζͷαϙʔτ
ݱঢ়ͷωοτػೳ ҎԼͷΞΠςϜΛ͏͜ͱͰ࣮ݱՄೳ ESP8266 or ESP32 ͱ͍͏֨҆Wi-Fiػೳ͖ϚΠίϯ driversʹ͋Δ ࠓճͷλʔήοτΛESPϞδϡʔϧͱUARTͱ͍͏̎৴߸ Λͭͳ͙ TCP/UDP௨৴͕ՄೳʹͳΔ
ESP8266/ESP32 AT Command set for WiFi/TCP/UDP
ࠓޙͷωοτଓ LINE ThingsͷΑ͏ʹࠓ୭͕࣋ͭεϚϗΛ BLEtoΠϯλʔωοτͷϋϒʹͪ͠Ό͏ΞΠσΟΞ LoRaɺNB-IoTϞδϡʔϧ΅ͪ΅ͪೖखՄೳʹͳΓͭͭ ͋Γ·͢
WebAssemblyग़ྗ ຊՈΑΓޮతͳՌΛग़ͤΔ syscall/jsαϙʔτͳͲ࢝·ͬͯΔ
IoT։ൃͲΜͲΜۙʹ ͳ͖ͬͯͯ·͢Αʂ
͓ΘΓ