// file      : libbuild2/cc/target.hxx -*- C++ -*-
// license   : MIT; see accompanying LICENSE file

#ifndef LIBBUILD2_CC_TARGET_HXX
#define LIBBUILD2_CC_TARGET_HXX

#include <libbuild2/types.hxx>
#include <libbuild2/utility.hxx>

#include <libbuild2/target.hxx>

#include <libbuild2/cc/export.hxx>

namespace build2
{
  namespace cc
  {
    // This is an abstract base target for all c-common header/source files.
    // We use this arrangement during rule matching to detect "unknown" (to
    // this rule) source/header files that it cannot handle but should not
    // ignore either. For example, a C link rule that sees a C++ source file.
    //
    class LIBBUILD2_CC_SYMEXPORT cc: public file
    {
    public:
      cc (context& c, dir_path d, dir_path o, string n)
        : file (c, move (d), move (o), move (n))
      {
        dynamic_type = &static_type;
      }

    public:
      static const target_type static_type;
    };

    // There is hardly a c-family compilation without a C header inclusion.
    // As a result, this target type is registered for any c-family module.
    //
    class LIBBUILD2_CC_SYMEXPORT h: public cc
    {
    public:
      h (context& c, dir_path d, dir_path o, string n)
        : cc (c, move (d), move (o), move (n))
      {
        dynamic_type = &static_type;
      }

    public:
      static const target_type static_type;
    };

    // This one we define in cc but the target type is only registered by the
    // c module. This way we can implement rule chaining without jumping
    // through too many hoops (like resolving target type dynamically) but
    // also without relaxing things too much (i.e., the user still won't be
    // able to refer to c{} without loading the c module).
    //
    class LIBBUILD2_CC_SYMEXPORT c: public cc
    {
    public:
      c (context& ctx, dir_path d, dir_path o, string n)
        : cc (ctx, move (d), move (o), move (n))
      {
        dynamic_type = &static_type;
      }

    public:
      static const target_type static_type;
    };

    // Objective-C source file (the same rationale for having it here as for
    // c{} above).
    //
    class LIBBUILD2_CC_SYMEXPORT m: public cc
    {
    public:
      m (context& c, dir_path d, dir_path o, string n)
        : cc (c, move (d), move (o), move (n))
      {
        dynamic_type = &static_type;
      }

    public:
      static const target_type static_type;
    };

    // Assembler with C preprocessor source file (the same rationale for
    // having it here as for c{} above).
    //
    class LIBBUILD2_CC_SYMEXPORT S: public cc
    {
    public:
      S (context& c, dir_path d, dir_path o, string n)
        : cc (c, move (d), move (o), move (n))
      {
        dynamic_type = &static_type;
      }

    public:
      static const target_type static_type;
    };

    // This is an abstract base target for deriving additional targets that
    // can be #include'd in C translation units (the same rationale for having
    // it here as for c{} above). In particular, only such targets will be
    // considered to reverse-lookup extensions to target types (see
    // dyndep_rule::map_extension() for background).
    //
    class LIBBUILD2_CC_SYMEXPORT c_inc: public cc
    {
    public:
      c_inc (context& c, dir_path d, dir_path o, string n)
        : cc (c, move (d), move (o), move (n))
      {
        dynamic_type = &static_type;
      }

    public:
      static const target_type static_type;
    };

    // pkg-config file targets.
    //
    class LIBBUILD2_CC_SYMEXPORT pc: public file // .pc (common)
    {
    public:
      pc (context& c, dir_path d, dir_path o, string n)
        : file (c, move (d), move (o), move (n))
      {
        dynamic_type = &static_type;
      }

    public:
      static const target_type static_type;
    };

    class LIBBUILD2_CC_SYMEXPORT pca: public pc // .static.pc
    {
    public:
      pca (context& c, dir_path d, dir_path o, string n)
        : pc (c, move (d), move (o), move (n))
      {
        dynamic_type = &static_type;
      }

    public:
      static const target_type static_type;
    };

    class LIBBUILD2_CC_SYMEXPORT pcs: public pc // .shared.pc
    {
    public:
      pcs (context& c, dir_path d, dir_path o, string n)
        : pc (c, move (d), move (o), move (n))
      {
        dynamic_type = &static_type;
      }

    public:
      static const target_type static_type;
    };
  }
}

#endif // LIBBUILD2_CC_TARGET_HXX