USB TERMAL PRINTER
2026.04.17

今回はUSBサーマルプリンタをマイコンから制御します。
TTLサーマルプリンタであれば、単にシリアル接続すれば済むのですが、
USBプリンタでは、USBシリアル変換モジュールを挟めば済むというものではありません。
USB HOST機能を有するESP32-S3から制御します。

使用するUSBサーマルプリンタは三栄電機株式会社のBL-112UI-BH、ファームウェアバージョン3.04、USB1.1準拠です。
PB-112PS/UI技術マニュアル
●BL-112UI プリンタ仕様

JIS第1・第2水準漢字は、日本工業規格(JIS X 0208)で定められたコンピュータ用漢字集合です。(6,879文字)

プリンタードライバー
ちなみに、Windows8 64bit版ドライバをWindows11にインストールしてみましたが、印刷に失敗してしまいました。
●ARDUINO NANO ESP32

USB HOST機能を実装しているESP32-S3によりUSBサーマルプリンタを制御してみます。
ESP32-S3であれば何でもよいわけではありません。GPIO19.GPIO20ピンが引き出されるマイコンが必要です。
●USBホスト機能

Nano ESP32 では、GPIO19(USB D-).GPIO20(USB D+)がUSB端子に引き出されており、USBホスト機能を利用することができます。
●USB PRINTライブラリ

touchgadget/esp32-usb-host-demos
こういったライブラリは、最新のUSBプリンタを認識できないことが多いのですが、
BL-112PS/UI取扱説明書の作成日付をみると2006年1月です。
この程度に古いプリンタ(USB1.1準拠)とは相性がよかったりします。
以下の3つのコードを使います。
- show_desc.hpp
- usbhhelp.hpp
- examples->usbhprinter->usbhprinter.ino
編集するコードは usbhprinter.ino です。
関数が呼び出される大まかな流れは下記の通りです。
static void printer_transfer_cb(usb_transfer_t *transfer)
{
ESP_LOGI("", "printer_transfer_cb context: %d", transfer->context);
~~~~~~~~~~~~~~~
}
void check_interface_desc_printer(const void *p)
{
~~~~~~~~~~~~~~~
ESP_LOGI("", "Claiming a %s-directional printer!",
(isBiDirectional)?"bi":"uni");
~~~~~~~~~~~~~~~
}
void prepare_endpoints(const void *p)
{
~~~~~~~~~~~~~~~
PrinterOut->callback = printer_transfer_cb;
~~~~~~~~~~~~~~~
}
void show_config_desc_full(const usb_config_desc_t *config_desc)
{
~~~~~~~~~~~~~~~
case USB_B_DESCRIPTOR_TYPE_INTERFACE:
show_interface_desc(p);
check_interface_desc_printer(p);
break;
case USB_B_DESCRIPTOR_TYPE_ENDPOINT:
show_endpoint_desc(p);
prepare_endpoints(p);
break;
~~~~~~~~~~~~~~~
}
void setup()
{
Serial.begin(115200);
Serial.setTimeout(0);
usbh_setup(show_config_desc_full);
}
void loop()
{
usbh_task();
String aLine = Serial.readStringUntil('\n');
if (aLine.length() > 0) {
aLine.concat('\n');
PrinterOut->num_bytes = aLine.length();
memcpy(PrinterOut->data_buffer,
aLine.c_str()
, PrinterOut->num_bytes);
esp_err_t err = usb_host_transfer_submit(PrinterOut);
if (err != ESP_OK) {
ESP_LOGI("", "usb_host_transfer_submit Out fail: %x", err);
}
}
}
まずはメッセージをSerial1に出力するように変更します
void setup()
{
Serial1.begin(115200,SERIAL_8N1, 6, 5);
Serial1.setTimeout(0);
usbh_setup(show_config_desc_full);
}
ESP32-S3では、Serialを任意のピンに割り振ることができるので、
ESP_LOGI()の出力先を明示的に指定します
ESP_LOGI("", ~)
↓
Serial1.printf(~)
ソースコードのビルドには、Raspberry Pi / PlatformIOを使用しています。
Arduino開発環境構築 PlatformIO

テストには月夜野4号車を使いました。

ロボットカー仕様になっていますがESP32-S3マイコンです。
ラズベリーパイとはビルド用にUSBケーブルを繋いでいます。
コードをビルドします
$ pio init -b arduino_nano_esp32
$ pio run -t upload

ビルド用に繋いでいたUSBケーブルを外して、プリンタと接続します。
実行時メッセージ表示用にラズベリーパイとシリアル接続します。
ESP32へは、ロボットカー搭載単4電池3個の電池ボックスから車体実装の3.3Vレギュレータを介して、VIN端子へ電力供給しています。
デバイスモニタを起動します。
$ pio device monitor -p /dev/ttyS0 -b 115200

サーマルプリンタの電源スイッチをオンにして、プログラムを起動するとメッセージが表示されます。

デバイスモニタ上からキーボード入力するたびに、コールバック関数が呼ばれ、関数内のメッセージ出力関数が実行されます。
Serial1.printf("printer_transfer_cb context: %d", transfer->context);
このメッセージは邪魔なので動作確認後はコメントアウトします。

1文字入力するたびにタイムアウトが発生するため即印字、改行されます。
印字は文字入力の際ではなく、改行コードが入力されると纏めて印字されます。
void setup()
{
Serial.begin(115200);
Serial.setTimeout(0);
usbh_setup(show_config_desc_full);
}
void loop()
{
usbh_task();
// ESP32 S2 Typewriter
// Read line from serial monitor and write to printer.
String aLine = Serial.readStringUntil('\n');
if (aLine.length() > 0) {
// readStringUntil removes the newline so add it back
aLine.concat('\n');
PrinterOut->num_bytes = aLine.length();
memcpy(PrinterOut->data_buffer,aLine.c_str(),PrinterOut->num_bytes);
esp_err_t err = usb_host_transfer_submit(PrinterOut);
●実用コードへの編集
1文字ごとに改行されては困るので、ソースコードを編集します。
char spool[SPOOL_BUFSIZE];
void setup()
{
Serial1.begin(115200,SERIAL_8N1, 6, 5);
//Serial1.setTimeout(0);
usbh_setup(show_config_desc_full);
}
void loop()
{
usbh_task();
if (Serial1.available()) {
spool[0] = Serial1.read();
Serial1.write(spool[0]);
PrinterOut->num_bytes = 1;
memcpy(PrinterOut->data_buffer, spool, PrinterOut->num_bytes);
esp_err_t err = usb_host_transfer_submit(PrinterOut);
if (err != ESP_OK) {
Serial1.printf("usb_host_transfer_submit Out fail: %x", err);
}
}
}
このコードでは入力文字のループバックも行っています。
Serial1.write(spool[0]);


●日本語処理
BL-112PS/UIの漢字コードはJIS X 0208です。
普段、テキストファイルはUTF-8で管理しているので、印刷時に漢字コードをJISに変換して印字しています。

Unicode対応 JIS X 0208 文字コード一覧
漢字印刷には前準備が必要です。
1. UTF-8をJISに変換するライブラリが見つからなかったので、対応表を空白区切りでLibreOfficeに取り込み、UTF-8とJISの1対1対応部分のみを抽出します。
2. 編集後にタブ区切りでUTF-8テキストファイルとして書き出します。
3. これをPHPで読み込んで、バイナリ形式でmicroSDカードに保存します。
[UTF-8(3バイト)][JIS(2バイト)]~[UTF-8(3バイト)][JIS(2バイト)]
ファイルサイズは5バイト掛ける6879文字で34395バイトになります。
プログラムへの実装についてです。
1. ESP32-S3起動時にバイナリファイルをメモリー上に読み込みます。
2. UTF-8の漢字コードが入力された際に、UTF-8コードを検索し、対応するJISコードを取得します。
3. 漢字モード開始コードをプリンタに送信、次にJISコードを出力、最後に漢字モード終了コードを伝えます。
[0x1c][0x26][JISコード(2バイト)][0x1c][0x2e]

24x24ドットの漢字表示ではくっきりと印字されています。
●さらなる拡張
パソコン上のTeraTermを起動して、ESP32へTELNET接続、文字入力やファイル送信により印刷できるようにしました。

ソースコードの場合は行番号を付けて印字しています。
英数字8x16、漢字16x16ドット、行間2ドットを指定しています。
BL-112UIプリンタ仕様にACアダプタ使用時の最大印字速度は640dot lines/secとあり、
縦16ドットのフォントを使用した場合1秒で40行(25ms/行)、24ドットフォントで26.7行(37.5ms/行)印刷が可能です。
印字速度よりESP32の処理速度が圧倒的に早いので、印刷命令usb_host_transfer_submit()を発行した後に、
delay(25)あるいはdelay(38)の休止を挟まないと、印字が追い付かずプリンタがバグります。
●印刷物の持ち運び

以前取り上げた、AS-289R2などのロール紙の幅は58mmで、スティックのりPiTなどの空き容器に収納できますが、BL-112PS/UIのロール紙の幅は112mmです。
そこで、明治のマーブルチョコの円筒を使いました。
|
Raspberry Pi(ラズベリー パイ)は、ARMプロセッサを搭載したシングルボードコンピュータ。イギリスのラズベリーパイ財団によって開発されている。
KEYESTUDIO Micro:bit V2ロボットスターターキット (マイクロビットなし) MakecodeとPython用
動き、光と音の使い方、障害物の検知と回避方法、線をたどり、IRリモコンとアプリでコントロールする方法を学びます。
Maqueen Lite V5 micro:bit ロボットキット STEM用(ライントレース&障害物回避)
Maqueen Lite V5 STEMロボットは、事前調整済みのライントレースアルゴリズム、耐衝撃モーター、開封即使用可能な設計で、ストレスを解消します。
たいていのことは100日あれば、うまくいく。長田英知著
「時間がなくて、なかなか自分のやりたいことができない」
「一念発起して何かを始めても、いつも三日坊主で終わってしまう」
「色んなことを先延ばしにしたまま、時間だけが過ぎていく」
そこで本書では、そんな著者が独自に開発した、
まったく新しい目標達成メソッド「100日デザイン」について、
その知識と技術を、余すところなくご紹介します。
まんがで納得ナポレオン・ヒル 思考は現実化する
OLとして雑務をこなす日々に飽き足らず、科学者だった父が残した薬品を商品化すべく、起業を決意した内山麻由(27)。彼女はセミナーで知り合った謎の女性からサポートを得ながら、彼女と二人三脚でナポレオン・ヒルの成功哲学を実践し、さまざまな問題を乗り越えていく。
ヒル博士の<ゴールデンルール>に従い、仕事に、恋に全力疾走する彼女の、成功への物語。
今日は人生最悪で最高の日 1秒で世界を変えるたったひとつの方法 ひすいこたろう著
偉人の伝記を読むと、最悪な日は、不幸な日ではなく、新しい自分が始まる日であることがわかります。最悪な出来事は、自分の人生が、想像を超えて面白くなる兆しなのです。偉人伝を読むことで、このときの不幸があったおかげで、未来にこういう幸せがくるのかと、人生を俯瞰する視線が立ち上がるのです。
ご飯は私を裏切らない heisoku著
辛い現実から目を背けて食べるご飯は、いつも美味しく幸せを届けてくれる。
29歳、中卒、恋人いない歴イコール年齢。バイト以外の職歴もなく、短期バイトを転々とする日々。ぐるぐると思索に耽るけど、ご飯を食べると幸せになれる。奇才の新鋭・heisokuが贈るリアル労働グルメ物語!
【最新版Gemini 3に対応!】できるGemini (できるシリーズ)
Geminiを「最強の知的生産パートナー」として使いこなすための、実践的なノウハウを凝縮した一冊です。
基本的な操作方法から、具体的なビジネスシーンでの活用、日々の業務を自動化するGoogle Workspaceとの連携、さらには自分だけのオリジナルAIを作成する方法まで余すところなく解説します。
Rustプログラミング完全ガイド 他言語との比較で違いが分かる!
Rustの各手法や考え方を幅広く解説!
500以上のサンプルを掲載。実行結果も確認。
全24章の包括的なチュートリアル。
ポチらせる文章術
販売サイト・ネット広告・メルマガ・ブログ・ホームページ・SNS…
全WEB媒体で効果バツグン!
カリスマコピーライターが教える「見てもらう」「買ってもらう」「共感してもらう」すべてに効くネット文章術
小型で便利な Type-C アダプター USB C オス - USB3.1 オスアダプター
Type-C端子のマイコンボードをこのアダプタを介して直接Raspberry Piに挿すことができます。ケーブルなしで便利なツールです。
Divoom Ditoo Pro ワイヤレススピーカー
15W高音質重低音/青軸キーボード/Bluetooth5.3/ピクセルアート 専用アプリ/USB接続/microSDカード
電源供給USBケーブル スリム 【5本セット】
USB電源ケーブル 5V DC電源供給ケーブル スリム 【5本セット】 電源供給 バッテリー 修理 自作 DIY 電子工作 (100cm)
|