İş mantığını uygulayın

  1. VSIDL kataloğu dizininizde gerekli derleme, protobuf ve VSIDL dosyalarının bulunduğunu doğrulayın. Mevcut bir örnekten başlamak için /system/software_defined_vehicle/samples/vsidl/stable/catalog altında geçerli bir katalog klasörü örneği bulabilirsiniz.

    Örnek katalog yapısı:

    my_catalog/
    ├── Android.bp          # Defines rust_protobuf modules for .proto files
    ├── types.proto         # Protobuf message / RPC service definitions
    └── architecture.vsidl  # VSIDL service bundle definitions
    
  2. İşletme mantığınızı uygulamak için özel kod yazın:

    • RPC sunucuları: lib.rs özelliklerini uygulayın.
    • Yayınlama, abone olma ve UPÇ istemcileri: Diğer hizmetlerle etkileşim kurmak için service_bundle.rs içinde oluşturulan işlevleri çağırın.

    Rust

    1. Aşağıdaki komutu çalıştırarak iskelet uygulama oluşturun:

      vsidlc -c /path/to/catalog -o /path/to/output --services
      

      vsidlc komutunu --services işaretiyle çalıştırmak, her hizmet paketi için standart bir Rust uygulaması oluşturur. Bu, kaynak kodu (main.rs) ve tüm bağımlılıkları içeren bir derleme yapılandırma dosyası (Android.bp) dahil olmak üzere gerekli iskeleti sağlar. Böylece, temel işletme mantığını uygulamaya odaklanırken varsayılan mesajların anında değiştirilmesine olanak tanır.

    2. Her hizmet paketi için Rust uygulamasını şu konumda bulabilirsiniz:

      /path/to/output/services/ServiceBundleName/src/main.rs.

    3. Oluşturulan uygulama, varsayılan olarak varsayılan değerlere sahip mesajlar oluşturur ve bunları yayıncılar ile aboneler veya RPC istemcileri ile sunucuları arasında gönderir. Bu davranışı değiştirmek için TODO bölümündeki yorumları main.rs bulun ve tercihlerinize göre ayarlayın. Örneğin:

      async fn handle_tire_pressure_range_unique_publisher(
          publisher: sdv::mw::Publisher<TirePressureRange>,
      ) {
          loop {
              // TODO: Modify the frequency of publishing messages here.
              sleep(Duration::from_secs(1)).await;
              // TODO: Modify the message content here.
              let message = TirePressureRange::default();
              info!("Publishing on TirePressureRange#UNIQUE");
              publisher.publish(&message).unwrap();
          }
      }
      
    4. Oluşturulan dosyalarda yaptığınız değişikliklerin vsidlc bir sonraki çalıştırılışında üzerine yazılmaması için services klasörünü /path/to/output klasörünün dışına taşıyın.

    C++

    1. Hizmet paketiniz için bir sınıf içeren yeni bir üst dosya oluşturun: src/lib.hpp,

      #pragma once
      #include <sdv/service_bundle.h>
      #include <sdv/context.hpp>
      
      namespace com::sdv::oem::service_bundle {
      
      // Sample implementation of the service bundle interface.
      class ServiceBundleName : public android::sdv::service_bundle::ServiceBundle {
      public:
          ServiceBundleName(sdv_comms::ctx::Context context);
          ~ServiceBundleName();
          void onStart() override;
          void onStop() override;
      };
      }  // namespace com::sdv::oem::service_bundle
      
    2. Sınıf uygulamasıyla birlikte, src/lib.cpp, yeni bir kaynak dosyası oluşturun:

      #include "src/lib.hpp"
      #include <sdv/sb_macro.h>
      #include <iostream>
      using com::sdv::oem::service_bundle::ServiceBundleName;
      
      // Register the new service bundle.
      REGISTER_SERVICE_BUNDLE(ServiceBundleName);
      
      // Sample implementation of the service bundle interface.
      namespace com::sdv::oem::service_bundle {
      // Creates a new instance of the ServiceBundleName.
      // Called when service bundle is created by the system.
      // Context object is provided as a parameter that gives access to the
      // communication stack APIs.
      ServiceBundleName::ServiceBundleName([[maybe_unused]] sdv_comms::ctx::Context context) :
        ServiceBundle(context) {
          // Memory allocations and static data loading should be done as
          // part of this method.
          //
          // Loading of the dynamic resources (sockets, files, etc)
          // is strongly discouraged due to possible Suspend-to-RAM scenario.
          //
          // The dynamic data can be loaded in the onStart method.
      }
      
      // Called when the service bundle is started by the system.
      void ServiceBundleName::onStart() {
          // Dynamic resources(sockets, files, etc) should be allocated during this call.
      }
      
      // Called when the service bundle is stopped by the system in preparation
      // for shutdown or suspend to RAM/Disc.
      void ServiceBundleName::onStop() {
          // Stop phase requires the service bundle to delete the dynamic resources
          // (sockets, files, etc) that were previously allocated in the onStart method.
      }
      
      // Called when the service bundle is destroyed by the system.
      ServiceBundleName::~ServiceBundleName() {
          // Static resources deallocation needs to be implemented in the destructor.
      }
      }  // namespace com::sdv::oem::service_bundle
      
    3. Yeni veya mevcut bir Android.bp dosyasında bir hizmet paketi kitaplığı Soong hedefi oluşturun:

      // Service bundle library.
      cc_library_shared {
        name: "libservice_bundle",
        srcs: ["src/lib.cpp",],
        // Allows the library to be available inside APEX.
        apex_available: [
            "//apex_available:platform",
            "//apex_available:anyapex",
        ],
        shared_libs: [
          // Service Bundle lifecycle C++ API.
          "libsdv_lifecycle_client_cpp",
          // commstack library that provides context object reference
          "libsdv_comms_cpp",
        ],
      
        // Service bundle package is packed into /product apexes.
        product_specific: true,
      }
      

Sonraki adımlar

Hizmet paketlerinizi dağıtmak için Hizmet paketleri oluşturma ve dağıtma başlıklı makaleyi inceleyin.