Work with more than 2 different structures in ASN.1 (C++) with asn1c compiler.

huangapple go评论87阅读模式
英文:

Work with more than 2 different structures in ASN.1 (C++) with asn1c compiler

问题

我在ASN.1中有两种不同的结构:一个矩形和一个圆:

  1. Mystructs DEFINITIONS ::= BEGIN
  2. Rectangle ::= SEQUENCE {
  3. height INTEGER (0..255),
  4. width INTEGER (0..255)
  5. }
  6. Circle ::= SEQUENCE {
  7. radius INTEGER (0..255)
  8. }
  9. END

在编译后,我可以对这些结构进行编码(uper_encode())并发送矩形或圆。我的问题是,当我想解码这些结构(uper_decode())时,我应该预先知道我正在接收什么,但实际上我不知道。我只知道它可能是一个圆或一个矩形。

因此,我想在发送编码数据后添加一个标签(如标头),以便在解码之前可以处理它,以知道应该使用哪个模板来解码:矩形还是圆。

但是,我不知道如何做到这一点。

谢谢!

英文:

I have 2 different structures in ASN.1: a rectangle and a circle:

  1. Mystructs DEFINITIONS ::= BEGIN
  2. Rectangle ::= SEQUENCE {
  3. height INTEGER (0..255),
  4. width INTEGER (0..255)
  5. }
  6. Circle ::= SEQUENCE {
  7. radius INTEGER (0..255)
  8. }
  9. END

After the compilation, I can encode these structures (uper_encode()) and send either the rectangle or the circle. My problem is that when I want to decode the structures (uper_decode()), I should know a priori what I am receiving but actually I don't know. I only know that it could be a circle or a rectangle.

So I thought that I could add somehow a tag after sending the encoded data (like a header) that I can process before I decode to know which template should I use to decode: rectangle or circle.

However, I do not know how to do this.

Thanks!!!

答案1

得分: 1

如@Botje和@Kevin所回答的,您需要一个CHOICE。

但是,如果您将其添加到规范中,您需要标记这些替代项(因为它们都是未标记的SEQUENCE类型)。

  1. Mystructs DEFINITIONS ::= BEGIN
  2. CircleOrRectangle ::= CHOICE {
  3. circle [0] Circle,
  4. rectangle [1] Rectangle
  5. }
  6. Rectangle ::= SEQUENCE {
  7. height INTEGER (0..255),
  8. width INTEGER (0..255)
  9. }
  10. Circle ::= SEQUENCE {
  11. radius INTEGER (0..255)
  12. }
  13. END

另一个选项,如果可以的话更好,是在DEFINITIONS级别添加AUTOMATIC TAGS。

  1. Mystructs DEFINITIONS AUTOMATIC TAGS ::= BEGIN
  2. CircleOrRectangle ::= CHOICE {
  3. circle Circle,
  4. rectangle Rectangle
  5. }
  6. Rectangle ::= SEQUENCE {
  7. height INTEGER (0..255),
  8. width INTEGER (0..255)
  9. }
  10. Circle ::= SEQUENCE {
  11. radius INTEGER (0..255)
  12. }
  13. END

编辑...

使用https://github.com/vlm/asn1c,这个CHOICE将在CircleOrRectangle.h中定义。

  1. /* 依赖项 */
  2. typedef enum CircleOrRectangle_PR {
  3. CircleOrRectangle_PR_NOTHING, /* 无组件 */
  4. CircleOrRectangle_PR_circle,
  5. CircleOrRectangle_PR_rectangle
  6. } CircleOrRectangle_PR;
  7. /* CircleOrRectangle */
  8. typedef struct CircleOrRectangle {
  9. CircleOrRectangle_PR present;
  10. union CircleOrRectangle_u {
  11. Circle_t circle;
  12. Rectangle_t rectangle;
  13. } choice;
  14. /* 跨越缓冲区边界的解析上下文 */
  15. asn_struct_ctx_t _asn_ctx;
  16. } CircleOrRectangle_t;

正如您所看到的,它被映射为一个包含枚举和联合的结构体。枚举将告诉您在联合中使用哪个成员。

英文:

As @Botje and @Kevin answered you need a CHOICE

However, if you add it to your specification, you need to tag the alternatives (because they are both untagged SEQUENCE types)

  1. Mystructs DEFINITIONS ::= BEGIN
  2. CircleOrRectangle ::= CHOICE {
  3. circle [0] Circle,
  4. rectangle [1] Rectangle
  5. }
  6. Rectangle ::= SEQUENCE {
  7. height INTEGER (0..255),
  8. width INTEGER (0..255)
  9. }
  10. Circle ::= SEQUENCE {
  11. radius INTEGER (0..255)
  12. }
  13. END

Another option, which is better if you can, is to add AUTOMATIC TAGS at the DEFINITIONS level

  1. Mystructs DEFINITIONS AUTOMATIC TAGS ::= BEGIN
  2. CircleOrRectangle ::= CHOICE {
  3. circle Circle,
  4. rectangle Rectangle
  5. }
  6. Rectangle ::= SEQUENCE {
  7. height INTEGER (0..255),
  8. width INTEGER (0..255)
  9. }
  10. Circle ::= SEQUENCE {
  11. radius INTEGER (0..255)
  12. }
  13. END

EDIT ...

Using https://github.com/vlm/asn1c, this choice will be defined in CircleOrRectangle.h

  1. /* Dependencies */
  2. typedef enum CircleOrRectangle_PR {
  3. CircleOrRectangle_PR_NOTHING, /* No components present */
  4. CircleOrRectangle_PR_circle,
  5. CircleOrRectangle_PR_rectangle
  6. } CircleOrRectangle_PR;
  7. /* CircleOrRectangle */
  8. typedef struct CircleOrRectangle {
  9. CircleOrRectangle_PR present;
  10. union CircleOrRectangle_u {
  11. Circle_t circle;
  12. Rectangle_t rectangle;
  13. } choice;
  14. /* Context for parsing across buffer boundaries */
  15. asn_struct_ctx_t _asn_ctx;
  16. } CircleOrRectangle_t;

As you see, it is mapped as as a struct containing an enum and a union. The enum will tell you which member to use in the union.

答案2

得分: 0

CircleOrRectangle ::= CHOICE {
circle 圆形,
rectangle 矩形
}

编码将携带关于所做选择的信息。

英文:

I think you are looking for a CHOICE type.

  1. CircleOrRectangle ::= CHOICE {
  2. circle Circle,
  3. rectangle Rectangle
  4. }

The encoding will then carry information about which selection was made.

huangapple
  • 本文由 发表于 2023年6月19日 19:40:12
  • 转载请务必保留本文链接:https://go.coder-hub.com/76506266.html
匿名

发表评论

匿名网友

:?: :razz: :sad: :evil: :!: :smile: :oops: :grin: :eek: :shock: :???: :cool: :lol: :mad: :twisted: :roll: :wink: :idea: :arrow: :neutral: :cry: :mrgreen:

确定