"python-package/vscode:/vscode.git/clone" did not exist on "1881a501782e0ab1e7618b3ab6ff8871ee311860"
serial_tree_learner.h 6.05 KB
Newer Older
Guolin Ke's avatar
Guolin Ke committed
1
2
3
4
5
6
7
8
9
#ifndef LIGHTGBM_TREELEARNER_SERIAL_TREE_LEARNER_H_
#define LIGHTGBM_TREELEARNER_SERIAL_TREE_LEARNER_H_

#include <LightGBM/utils/random.h>
#include <LightGBM/utils/array_args.h>

#include <LightGBM/tree_learner.h>
#include <LightGBM/dataset.h>
#include <LightGBM/tree.h>
Guolin Ke's avatar
Guolin Ke committed
10

Guolin Ke's avatar
Guolin Ke committed
11
12
#include "feature_histogram.hpp"
#include "split_info.hpp"
Guolin Ke's avatar
Guolin Ke committed
13
#include "data_partition.hpp"
Guolin Ke's avatar
Guolin Ke committed
14
15
16
17
18
19
#include "leaf_splits.hpp"

#include <cstdio>
#include <vector>
#include <random>
#include <cmath>
Guolin Ke's avatar
Guolin Ke committed
20
#include <memory>
21
22
23
24
25
#ifdef USE_GPU
// Use 4KBytes aligned allocator for ordered gradients and ordered hessians when GPU is enabled.
// This is necessary to pin the two arrays in memory and make transferring faster.
#include <boost/align/aligned_allocator.hpp>
#endif
Guolin Ke's avatar
Guolin Ke committed
26
27
28
29
30
31
32
33

namespace LightGBM {

/*!
* \brief Used for learning a tree by single machine
*/
class SerialTreeLearner: public TreeLearner {
public:
Guolin Ke's avatar
Guolin Ke committed
34
  explicit SerialTreeLearner(const TreeConfig* tree_config);
Guolin Ke's avatar
Guolin Ke committed
35
36
37

  ~SerialTreeLearner();

38
  void Init(const Dataset* train_data, bool is_constant_hessian) override;
Guolin Ke's avatar
Guolin Ke committed
39

Guolin Ke's avatar
Guolin Ke committed
40
41
  void ResetTrainingData(const Dataset* train_data) override;

Guolin Ke's avatar
Guolin Ke committed
42
43
  void ResetConfig(const TreeConfig* tree_config) override;

44
  Tree* Train(const score_t* gradients, const score_t *hessians, bool is_constant_hessian) override;
Guolin Ke's avatar
Guolin Ke committed
45

46
  Tree* FitByExistingTree(const Tree* old_tree, const score_t* gradients, const score_t* hessians) const override;
Guolin Ke's avatar
Guolin Ke committed
47

48
49
50
  Tree* FitByExistingTree(const Tree* old_tree, const std::vector<int>& leaf_pred,
                          const score_t* gradients, const score_t* hessians) override;

Guolin Ke's avatar
Guolin Ke committed
51
52
53
54
  void SetBaggingData(const data_size_t* used_indices, data_size_t num_data) override {
    data_partition_->SetUsedDataIndices(used_indices, num_data);
  }

Guolin Ke's avatar
Guolin Ke committed
55
56
57
  void AddPredictionToScore(const Tree* tree, double* out_score) const override {
    if (tree->num_leaves() <= 1) { return; }
    CHECK(tree->num_leaves() <= data_partition_->num_leaves());
Guolin Ke's avatar
Guolin Ke committed
58
    #pragma omp parallel for schedule(static)
59
    for (int i = 0; i < tree->num_leaves(); ++i) {
Guolin Ke's avatar
Guolin Ke committed
60
      double output = static_cast<double>(tree->LeafOutput(i));
Guolin Ke's avatar
Guolin Ke committed
61
62
      data_size_t cnt_leaf_data = 0;
      auto tmp_idx = data_partition_->GetIndexOnLeaf(i, &cnt_leaf_data);
Guolin Ke's avatar
Guolin Ke committed
63
      for (data_size_t j = 0; j < cnt_leaf_data; ++j) {
64
        out_score[tmp_idx[j]] += output;
Guolin Ke's avatar
Guolin Ke committed
65
66
67
68
69
70
71
72
73
74
75
76
77
      }
    }
  }

protected:
  /*!
  * \brief Some initial works before training
  */
  virtual void BeforeTrain();

  /*!
  * \brief Some initial works before FindBestSplit
  */
Guolin Ke's avatar
Guolin Ke committed
78
  virtual bool BeforeFindBestSplit(const Tree* tree, int left_leaf, int right_leaf);
Guolin Ke's avatar
Guolin Ke committed
79

Guolin Ke's avatar
Guolin Ke committed
80
  virtual void FindBestSplits();
Guolin Ke's avatar
Guolin Ke committed
81

Guolin Ke's avatar
Guolin Ke committed
82
  virtual void ConstructHistograms(const std::vector<int8_t>& is_feature_used, bool use_subtract);
Guolin Ke's avatar
Guolin Ke committed
83

Guolin Ke's avatar
Guolin Ke committed
84
  virtual void FindBestSplitsFromHistograms(const std::vector<int8_t>& is_feature_used, bool use_subtract);
Guolin Ke's avatar
Guolin Ke committed
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107

  /*!
  * \brief Partition tree and data according best split.
  * \param tree Current tree, will be splitted on this function.
  * \param best_leaf The index of leaf that will be splitted.
  * \param left_leaf The index of left leaf after splitted.
  * \param right_leaf The index of right leaf after splitted.
  */
  virtual void Split(Tree* tree, int best_leaf, int* left_leaf, int* right_leaf);

  /*!
  * \brief Get the number of data in a leaf
  * \param leaf_idx The index of leaf
  * \return The number of data in the leaf_idx leaf
  */
  inline virtual data_size_t GetGlobalDataCountInLeaf(int leaf_idx) const;
  /*! \brief number of data */
  data_size_t num_data_;
  /*! \brief number of features */
  int num_features_;
  /*! \brief training data */
  const Dataset* train_data_;
  /*! \brief gradients of current iteration */
108
  const score_t* gradients_;
Guolin Ke's avatar
Guolin Ke committed
109
  /*! \brief hessians of current iteration */
110
  const score_t* hessians_;
Guolin Ke's avatar
Guolin Ke committed
111
  /*! \brief training data partition on leaves */
Guolin Ke's avatar
Guolin Ke committed
112
  std::unique_ptr<DataPartition> data_partition_;
Guolin Ke's avatar
Guolin Ke committed
113
114
  /*! \brief used for generate used features */
  Random random_;
Hui Xue's avatar
Hui Xue committed
115
  /*! \brief used for sub feature training, is_feature_used_[i] = false means don't used feature i */
Guolin Ke's avatar
Guolin Ke committed
116
  std::vector<int8_t> is_feature_used_;
117
118
  /*! \brief pointer to histograms array of parent of current leaves */
  FeatureHistogram* parent_leaf_histogram_array_;
Guolin Ke's avatar
Guolin Ke committed
119
120
121
122
123
124
125
126
127
  /*! \brief pointer to histograms array of smaller leaf */
  FeatureHistogram* smaller_leaf_histogram_array_;
  /*! \brief pointer to histograms array of larger leaf */
  FeatureHistogram* larger_leaf_histogram_array_;

  /*! \brief store best split points for all leaves */
  std::vector<SplitInfo> best_split_per_leaf_;

  /*! \brief stores best thresholds for all feature for smaller leaf */
Guolin Ke's avatar
Guolin Ke committed
128
  std::unique_ptr<LeafSplits> smaller_leaf_splits_;
Guolin Ke's avatar
Guolin Ke committed
129
  /*! \brief stores best thresholds for all feature for larger leaf */
Guolin Ke's avatar
Guolin Ke committed
130
  std::unique_ptr<LeafSplits> larger_leaf_splits_;
131
  std::vector<int> valid_feature_indices_;
Guolin Ke's avatar
Guolin Ke committed
132

133
134
#ifdef USE_GPU
  /*! \brief gradients of current iteration, ordered for cache optimized, aligned to 4K page */
135
  std::vector<score_t, boost::alignment::aligned_allocator<score_t, 4096>> ordered_gradients_;
136
  /*! \brief hessians of current iteration, ordered for cache optimized, aligned to 4K page */
137
  std::vector<score_t, boost::alignment::aligned_allocator<score_t, 4096>> ordered_hessians_;
138
#else
Guolin Ke's avatar
Guolin Ke committed
139
  /*! \brief gradients of current iteration, ordered for cache optimized */
140
  std::vector<score_t> ordered_gradients_;
Guolin Ke's avatar
Guolin Ke committed
141
  /*! \brief hessians of current iteration, ordered for cache optimized */
142
  std::vector<score_t> ordered_hessians_;
143
#endif
Guolin Ke's avatar
Guolin Ke committed
144
145

  /*! \brief Store ordered bin */
Guolin Ke's avatar
Guolin Ke committed
146
  std::vector<std::unique_ptr<OrderedBin>> ordered_bins_;
Guolin Ke's avatar
Guolin Ke committed
147
148
149
  /*! \brief True if has ordered bin */
  bool has_ordered_bin_ = false;
  /*! \brief  is_data_in_leaf_[i] != 0 means i-th data is marked */
Guolin Ke's avatar
Guolin Ke committed
150
  std::vector<char> is_data_in_leaf_;
151
  /*! \brief used to cache historical histogram to speed up*/
Guolin Ke's avatar
Guolin Ke committed
152
  HistogramPool histogram_pool_;
Guolin Ke's avatar
Guolin Ke committed
153
  /*! \brief config of tree learner*/
Guolin Ke's avatar
Guolin Ke committed
154
  const TreeConfig* tree_config_;
Guolin Ke's avatar
Guolin Ke committed
155
  int num_threads_;
Guolin Ke's avatar
Guolin Ke committed
156
  std::vector<int> ordered_bin_indices_;
157
  bool is_constant_hessian_;
Guolin Ke's avatar
Guolin Ke committed
158
159
};

Luke Gallagher's avatar
Luke Gallagher committed
160
161
162
inline data_size_t SerialTreeLearner::GetGlobalDataCountInLeaf(int leaf_idx) const {
  if (leaf_idx >= 0) {
    return data_partition_->leaf_count(leaf_idx);
Guolin Ke's avatar
Guolin Ke committed
163
164
165
166
167
168
  } else {
    return 0;
  }
}

}  // namespace LightGBM
Guolin Ke's avatar
Guolin Ke committed
169
#endif   // LightGBM_TREELEARNER_SERIAL_TREE_LEARNER_H_