質問をすることでしか得られない、回答やアドバイスがある。

15分調べてもわからないことは、質問しよう!

ただいまの
回答率

87.33%

ポインタ配列のPROGMEMの要素を変数を使って出したい

解決済

回答 2

投稿 編集

  • 評価
  • クリップ 0
  • VIEW 1,269

score 4

前提・実現したいこと

Arduinoのプログラムで、それぞれの配列の長さが違うのでポインタの配列を変数としてPROGMEM機能を使った際に思った通りになりませんでした。
どうすれば上手くいくのか、できないのであれば理由を教えていただけると幸いです。
下記のプログラムは例ですが、同じ問題が起こります。

ソースコード

const int Array1[] PROGMEM = {1, 2};
const int Array2[] PROGMEM = {4, 8};
const int Array3[] PROGMEM = {16, 32};
const int Array4[] PROGMEM = {64, 128};
int *const Length[] PROGMEM = {Array1, Array2, Array3, Array4};

void setup() {
  Serial.begin(9600);
}

void loop() {
  for (int i = 0; i < 4; i++) {
    for (int j = 0; j < 2; j++) {
      Serial.println(pgm_read_byte(&(Length[i][j])));
    }
  }
  while (1);
}

出力

12  //1のはずが…
65  //2のはずが…
239 //4のはずが…
224 //8のはずが…
12  //16のはずが…
65  //32のはずが…
12  //64のはずが…
65  //128のはずが…

試したこと

直接指定してみる
const int Array1[] PROGMEM = {1, 2};
const int Array2[] PROGMEM = {4, 8};
const int Array3[] PROGMEM = {16, 32};
const int Array4[] PROGMEM = {64, 128};
int *const Length[] PROGMEM = {Array1, Array2, Array3, Array4};

int const Length1[][2] PROGMEM = {
  {1, 2},
  {4, 8},
  {16, 32},
  {64, 128}
};

void setup() {
  Serial.begin(9600);
}

void loop() {
  Serial.println(pgm_read_byte(&(Length[0][0])));
  Serial.println(pgm_read_byte(&(Length[0][1])));
  Serial.println(pgm_read_byte(&(Length[1][0])));
  Serial.println(pgm_read_byte(&(Length[1][1])));
  Serial.println(pgm_read_byte(&(Length[2][0])));
  Serial.println(pgm_read_byte(&(Length[2][1])));
  Serial.println(pgm_read_byte(&(Length[3][0])));
  Serial.println(pgm_read_byte(&(Length[3][1])));
  while (1);
}
1
2
4
8
16
32
64
128//これだとうまくいくのはなぜなのか

補足情報

Arduino 1.8.13
Arduino Uno を使っています。
よろしくお願いします。

  • 気になる質問をクリップする

    クリップした質問は、後からいつでもマイページで確認できます。

    またクリップした質問に回答があった際、通知やメールを受け取ることができます。

    クリップを取り消します

  • 良い質問の評価を上げる

    以下のような質問は評価を上げましょう

    • 質問内容が明確
    • 自分も答えを知りたい
    • 質問者以外のユーザにも役立つ

    評価が高い質問は、TOPページの「注目」タブのフィードに表示されやすくなります。

    質問の評価を上げたことを取り消します

  • 評価を下げられる数の上限に達しました

    評価を下げることができません

    • 1日5回まで評価を下げられます
    • 1日に1ユーザに対して2回まで評価を下げられます

    質問の評価を下げる

    teratailでは下記のような質問を「具体的に困っていることがない質問」、「サイトポリシーに違反する質問」と定義し、推奨していません。

    • プログラミングに関係のない質問
    • やってほしいことだけを記載した丸投げの質問
    • 問題・課題が含まれていない質問
    • 意図的に内容が抹消された質問
    • 過去に投稿した質問と同じ内容の質問
    • 広告と受け取られるような投稿

    評価が下がると、TOPページの「アクティブ」「注目」タブのフィードに表示されにくくなります。

    質問の評価を下げたことを取り消します

    この機能は開放されていません

    評価を下げる条件を満たしてません

    評価を下げる理由を選択してください

    詳細な説明はこちら

    上記に当てはまらず、質問内容が明確になっていない質問には「情報の追加・修正依頼」機能からコメントをしてください。

    質問の評価を下げる機能の利用条件

    この機能を利用するためには、以下の事項を行う必要があります。

回答 2

checkベストアンサー

+1

「そういうもの」で満足ですか?

Length[]の宣言がconst int *const Length[] PROGMEM = {Array1, Array2, Array3, Array4};
配列がconst int[]なので読み出しがpgm_read_word()なのはともかく。

ソースコードをコンパイルしてアセンブルレベルで覗いてみたところ、

pgm_read_word(&Length[0][1])等のように直接指定した場合は、コンパイル時に定数に展開しています。つまり、ソースコード上&Length[0][1]と記述していても、これはLength[0]は定数Array1、&Length[0][1]は定数Array1+1としてコンパイルされています。結果、prg_read_word(&Length[0][1])は2を読み出すことになります。

一方、
pgm_read_word(&Length[i][j])と記述した場合は、Length[i]をRAMエリアに配置された配列(これを仮に変数(RAM)と表す)として*(Length(RAM)+i)の変数として求め、さらに*(Length(RAM)+i)(RAM)+jを今度はプログラムエリアのアドレスとして値を求める、ということになるようです。

ということで、
&Length[i]をプログラム領域として処理するよう明示的に指示してやる必要があって、結果
Serial.print(pgm_read_word(&((int*)pgm_read_ptr(&Length[i]))[j]));

1
2
4
8
<以下略>


が得られます。

投稿

  • 回答の評価を上げる

    以下のような回答は評価を上げましょう

    • 正しい回答
    • わかりやすい回答
    • ためになる回答

    評価が高い回答ほどページの上位に表示されます。

  • 回答の評価を下げる

    下記のような回答は推奨されていません。

    • 間違っている回答
    • 質問の回答になっていない投稿
    • スパムや攻撃的な表現を用いた投稿

    評価を下げる際はその理由を明確に伝え、適切な回答に修正してもらいましょう。

  • 2020/07/26 18:15

    回答していただき、ありがとうございます。
    なるほど、直接指定した場合には勝手に置き換わっているのですね、納得しました。
    そしてプログラム領域から読み取るpgm_read_ptr()というものは知りませんでした。
    疑問が解消しました、ありがとうございます!

    キャンセル

  • 2020/07/26 18:24

    pgm_read_ptr()は私も知っていたわけではないのですが、この件で検索していると出てくるavr/pgmspace.hを覗いていたらあったので使ってみました。

    あなたにとって「ベスト」であった回答をベストアンサーに選んで下さいね。一度選んだベストアンサーの取り消し/選び直しも出来るようになっています。

    キャンセル

0

AVR(ATMega328)では、フラッシュメモリとRAMは独立していて別のメモリとしてアクセスされます
PROGMEMを指すポインタ/アドレスは使用できません。

Arduino 日本語リファレンス

投稿

  • 回答の評価を上げる

    以下のような回答は評価を上げましょう

    • 正しい回答
    • わかりやすい回答
    • ためになる回答

    評価が高い回答ほどページの上位に表示されます。

  • 回答の評価を下げる

    下記のような回答は推奨されていません。

    • 間違っている回答
    • 質問の回答になっていない投稿
    • スパムや攻撃的な表現を用いた投稿

    評価を下げる際はその理由を明確に伝え、適切な回答に修正してもらいましょう。

  • 2020/07/25 08:42 編集

    回答していただき、ありがとうございます。
    やはりきちんと配列の配列で宣言するほかなさそうですね。
    ではどうして直接指定した場合にはうまく働いているように見えるのでしょうか、教えていただけると幸いです。

    キャンセル

  • 2020/07/25 08:45

    そういうもの、としかいいようがないです
    コンパイラはGCCなんだから、ご自分でポインタ使えるように修正すればどうでしょう。

    キャンセル

  • 2020/07/25 09:04

    そういうものなんですか…がんばってみます。

    キャンセル

15分調べてもわからないことは、teratailで質問しよう!

  • ただいまの回答率 87.33%
  • 質問をまとめることで、思考を整理して素早く解決
  • テンプレート機能で、簡単に質問をまとめられる

関連した質問

同じタグがついた質問を見る