/* Copyright 2016 The TensorFlow Authors. All Rights Reserved. Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with the License. You may obtain a copy of the License at http://www.apache.org/licenses/LICENSE-2.0 Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the specific language governing permissions and limitations under the License. ==============================================================================*/ #ifndef TENSORFLOW_CORE_KERNELS_SDCA_INTERNAL_H_ #define TENSORFLOW_CORE_KERNELS_SDCA_INTERNAL_H_ #define EIGEN_USE_THREADS #include #include #include #include #include #include #include #include #include "third_party/eigen3/unsupported/Eigen/CXX11/Tensor" #include "tensorflow/core/framework/device_base.h" #include "tensorflow/core/framework/op_kernel.h" #include "tensorflow/core/framework/tensor.h" #include "tensorflow/core/framework/tensor_shape.h" #include "tensorflow/core/framework/tensor_types.h" #include "tensorflow/core/framework/types.h" #include "tensorflow/core/kernels/loss.h" #include "tensorflow/core/lib/core/coding.h" #include "tensorflow/core/lib/core/errors.h" #include "tensorflow/core/lib/core/status.h" #include "tensorflow/core/lib/core/stringpiece.h" #include "tensorflow/core/lib/gtl/inlined_vector.h" #include "tensorflow/core/lib/random/distribution_sampler.h" #include "tensorflow/core/lib/strings/stringprintf.h" #include "tensorflow/core/util/guarded_philox_random.h" #include "tensorflow/core/util/work_sharder.h" namespace tensorflow { namespace sdca { // Statistics computed with input (ModelWeights, Example). struct ExampleStatistics { // Logits for each class. // For binary case, this should be a vector of length 1; while for multiclass // case, this vector has the same length as the number of classes, where each // value corresponds to one class. // Use InlinedVector to avoid heap allocation for small number of classes. gtl::InlinedVector wx; // Logits for each class, using the previous weights. gtl::InlinedVector prev_wx; // Sum of squared feature values occurring in the example divided by // L2 * sum(example_weights). double normalized_squared_norm = 0; // Num_weight_vectors equals to the number of classification classes in the // multiclass case; while for binary case, it is 1. ExampleStatistics(const int num_weight_vectors) : wx(num_weight_vectors, 0.0), prev_wx(num_weight_vectors, 0.0) {} }; class Regularizations { public: Regularizations() {} // Initialize() must be called immediately after construction. Status Initialize(OpKernelConstruction* const context) { TF_RETURN_IF_ERROR(context->GetAttr("l1", &symmetric_l1_)); TF_RETURN_IF_ERROR(context->GetAttr("l2", &symmetric_l2_)); shrinkage_ = symmetric_l1_ / symmetric_l2_; return Status::OK(); } // Proximal SDCA shrinking for L1 regularization. double Shrink(const double weight) const { const double shrinked = std::max(std::abs(weight) - shrinkage_, 0.0); if (shrinked > 0.0) { return std::copysign(shrinked, weight); } return 0.0; } // Vectorized float variant of the above. Eigen::Tensor EigenShrinkVector( const Eigen::Tensor weights) const { // Proximal step on the weights which is sign(w)*|w - shrinkage|+. return weights.sign() * ((weights.abs() - weights.constant(shrinkage_)) .cwiseMax(weights.constant(0.0))); } // Matrix float variant of the above. Eigen::Tensor EigenShrinkMatrix( const Eigen::Tensor weights) const { // Proximal step on the weights which is sign(w)*|w - shrinkage|+. return weights.sign() * ((weights.abs() - weights.constant(shrinkage_)) .cwiseMax(weights.constant(0.0))); } float symmetric_l2() const { return symmetric_l2_; } private: float symmetric_l1_ = 0; float symmetric_l2_ = 0; // L1 divided by L2, pre-computed for use during weight shrinking. double shrinkage_ = 0; TF_DISALLOW_COPY_AND_ASSIGN(Regularizations); }; class ModelWeights; // Struct describing a single example. class Example { public: // Compute matrix vector product between weights (a matrix) and features // (a vector). This method also computes the normalized example norm used // in SDCA update. // For multiclass case, num_weight_vectors equals to the number of classes; // while for binary case, it is 1. const ExampleStatistics ComputeWxAndWeightedExampleNorm( const int num_loss_partitions, const ModelWeights& model_weights, const Regularizations& regularization, const int num_weight_vectors) const; float example_label() const { return example_label_; } float example_weight() const { return example_weight_; } double squared_norm() const { return squared_norm_; } // Sparse features associated with the example. // Indices and Values are the associated feature index, and values. Values // can be optionally absent, in which we case we implicitly assume a value of // 1.0f. struct SparseFeatures { std::unique_ptr::UnalignedConstVec> indices; std::unique_ptr::UnalignedConstVec> values; // nullptr encodes optional. }; // A dense vector which is a row-slice of the underlying matrix. struct DenseVector { // Returns a row slice from the matrix. Eigen::TensorMap> Row() const { return Eigen::TensorMap>( data_matrix.data() + row_index * data_matrix.dimension(1), data_matrix.dimension(1)); } // Returns a row slice as a 1 * F matrix, where F is the number of features. Eigen::TensorMap> RowAsMatrix() const { return Eigen::TensorMap>( data_matrix.data() + row_index * data_matrix.dimension(1), 1, data_matrix.dimension(1)); } const TTypes::ConstMatrix data_matrix; const int64 row_index; }; private: std::vector sparse_features_; std::vector> dense_vectors_; float example_label_ = 0; float example_weight_ = 0; double squared_norm_ = 0; // sum squared norm of the features. // Examples fills Example in a multi-threaded way. friend class Examples; // ModelWeights use each example for model update w += \alpha * x_{i}; friend class ModelWeights; }; // Weights related to features. For example, say you have two sets of sparse // features i.e. age bracket and country, then FeatureWeightsDenseStorage hold // the parameters for it. We keep track of the original weight passed in and the // delta weight which the optimizer learns in each call to the optimizer. class FeatureWeightsDenseStorage { public: FeatureWeightsDenseStorage(const TTypes::Matrix nominals, TTypes::Matrix deltas) : nominals_(nominals), deltas_(deltas) { CHECK_GT(deltas.rank(), 1); } // Check if a feature index is with-in the bounds. bool IndexValid(const int64 index) const { return index >= 0 && index < deltas_.dimension(1); } // Nominals here are the original weight matrix. TTypes::Matrix nominals() const { return nominals_; } // Delta weights durining mini-batch updates. TTypes::Matrix deltas() const { return deltas_; } // Updates delta weights based on active dense features in the example and // the corresponding dual residual. void UpdateDenseDeltaWeights( const Eigen::ThreadPoolDevice& device, const Example::DenseVector& dense_vector, const std::vector& normalized_bounded_dual_delta); private: // The nominal value of the weight for a feature (indexed by its id). const TTypes::Matrix nominals_; // The accumulated delta weight for a feature (indexed by its id). TTypes::Matrix deltas_; }; // Similar to FeatureWeightsDenseStorage, but the underlying weights are stored // in an unordered map. class FeatureWeightsSparseStorage { public: FeatureWeightsSparseStorage(const TTypes::Vec indices, const TTypes::Matrix nominals, TTypes::Matrix deltas) : nominals_(nominals), deltas_(deltas) { // Create a map from sparse index to the dense index of the underlying // storage. for (int64 j = 0; j < indices.size(); ++j) { indices_to_id_[indices(j)] = j; } } // Check if a feature index exists. bool IndexValid(const int64 index) const { return indices_to_id_.find(index) != indices_to_id_.end(); } // Nominal value at a particular feature index and class label. float nominals(const int class_id, const int64 index) const { auto it = indices_to_id_.find(index); return nominals_(class_id, it->second); } // Delta weights durining mini-batch updates. float deltas(const int class_id, const int64 index) const { auto it = indices_to_id_.find(index); return deltas_(class_id, it->second); } // Updates delta weights based on active sparse features in the example and // the corresponding dual residual. void UpdateSparseDeltaWeights( const Eigen::ThreadPoolDevice& device, const Example::SparseFeatures& sparse_features, const std::vector& normalized_bounded_dual_delta); private: // The nominal value of the weight for a feature (indexed by its id). const TTypes::Matrix nominals_; // The accumulated delta weight for a feature (indexed by its id). TTypes::Matrix deltas_; // Map from feature index to an index to the dense vector. std::unordered_map indices_to_id_; }; // Weights in the model, wraps both current weights, and the delta weights // for both sparse and dense features. class ModelWeights { public: ModelWeights() {} bool SparseIndexValid(const int col, const int64 index) const { return sparse_weights_[col].IndexValid(index); } bool DenseIndexValid(const int col, const int64 index) const { return dense_weights_[col].IndexValid(index); } // Go through all the features present in the example, and update the // weights based on the dual delta. void UpdateDeltaWeights( const Eigen::ThreadPoolDevice& device, const Example& example, const std::vector& normalized_bounded_dual_delta); Status Initialize(OpKernelContext* const context); const std::vector& sparse_weights() const { return sparse_weights_; } const std::vector& dense_weights() const { return dense_weights_; } private: std::vector sparse_weights_; std::vector dense_weights_; TF_DISALLOW_COPY_AND_ASSIGN(ModelWeights); }; // Examples contains all the training examples that SDCA uses for a mini-batch. class Examples { public: Examples() {} // Returns the Example at |example_index|. const Example& example(const int example_index) const { return examples_.at(example_index); } int sampled_index(const int id) const { return sampled_index_[id]; } // Adaptive SDCA in the current implementation only works for // binary classification, where the input argument for num_weight_vectors // is 1. Status SampleAdaptiveProbabilities( const int num_loss_partitions, const Regularizations& regularization, const ModelWeights& model_weights, const TTypes::Matrix example_state_data, const std::unique_ptr& loss_updater, const int num_weight_vectors); void RandomShuffle(); int num_examples() const { return examples_.size(); } int num_features() const { return num_features_; } // Initialize() must be called immediately after construction. Status Initialize(OpKernelContext* const context, const ModelWeights& weights, int num_sparse_features, int num_sparse_features_with_values, int num_dense_features); private: // Reads the input tensors, and builds the internal representation for sparse // features per example. This function modifies the |examples| passed in // to build the sparse representations. static Status CreateSparseFeatureRepresentation( const DeviceBase::CpuWorkerThreads& worker_threads, int num_examples, int num_sparse_features, const ModelWeights& weights, const OpInputList& sparse_example_indices_inputs, const OpInputList& sparse_feature_indices_inputs, const OpInputList& sparse_feature_values_inputs, std::vector* const examples); // Reads the input tensors, and builds the internal representation for dense // features per example. This function modifies the |examples| passed in // to build the sparse representations. static Status CreateDenseFeatureRepresentation( const DeviceBase::CpuWorkerThreads& worker_threads, int num_examples, int num_dense_features, const ModelWeights& weights, const OpInputList& dense_features_inputs, std::vector* const examples); // Computes squared example norm per example i.e |x|^2. This function modifies // the |examples| passed in and adds the squared norm per example. static Status ComputeSquaredNormPerExample( const DeviceBase::CpuWorkerThreads& worker_threads, int num_examples, int num_sparse_features, int num_dense_features, std::vector* const examples); // All examples in the batch. std::vector examples_; // Adaptive sampling variables. std::vector probabilities_; std::vector sampled_index_; std::vector sampled_count_; int num_features_ = 0; TF_DISALLOW_COPY_AND_ASSIGN(Examples); }; } // namespace sdca } // namespace tensorflow #endif // TENSORFLOW_CORE_KERNELS_SDCA_INTERNAL_H_