1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. 2 // Use of this source code is governed by a BSD-style license that can be 3 // found in the LICENSE file. 4 5 #include "chrome/browser/sync/glue/tab_node_pool.h" 6 7 #include "base/format_macros.h" 8 #include "base/logging.h" 9 #include "base/strings/stringprintf.h" 10 #include "base/strings/utf_string_conversions.h" 11 #include "chrome/browser/sync/profile_sync_service.h" 12 #include "sync/internal_api/public/base/model_type.h" 13 #include "sync/internal_api/public/read_node.h" 14 #include "sync/internal_api/public/write_node.h" 15 #include "sync/internal_api/public/write_transaction.h" 16 17 namespace browser_sync { 18 19 static const char kNoSessionsFolderError[] = 20 "Server did not create the top-level sessions node. We " 21 "might be running against an out-of-date server."; 22 23 const size_t TabNodePool::kFreeNodesLowWatermark = 25; 24 const size_t TabNodePool::kFreeNodesHighWatermark = 100; 25 26 TabNodePool::TabNodePool(ProfileSyncService* sync_service) 27 : max_used_tab_node_id_(kInvalidTabNodeID), 28 sync_service_(sync_service) {} 29 30 // static 31 // We start vending tab node IDs at 0. 32 const int TabNodePool::kInvalidTabNodeID = -1; 33 34 TabNodePool::~TabNodePool() {} 35 36 // Static 37 std::string TabNodePool::TabIdToTag( 38 const std::string machine_tag, int tab_node_id) { 39 return base::StringPrintf("%s %d", machine_tag.c_str(), tab_node_id); 40 } 41 42 void TabNodePool::AddTabNode(int tab_node_id) { 43 DCHECK_GT(tab_node_id, kInvalidTabNodeID); 44 DCHECK(nodeid_tabid_map_.find(tab_node_id) == nodeid_tabid_map_.end()); 45 unassociated_nodes_.insert(tab_node_id); 46 if (max_used_tab_node_id_ < tab_node_id) 47 max_used_tab_node_id_ = tab_node_id; 48 } 49 50 void TabNodePool::AssociateTabNode(int tab_node_id, 51 SessionID::id_type tab_id) { 52 DCHECK_GT(tab_node_id, kInvalidTabNodeID); 53 // Remove sync node if it is in unassociated nodes pool. 54 std::set<int>::iterator u_it = unassociated_nodes_.find(tab_node_id); 55 if (u_it != unassociated_nodes_.end()) { 56 unassociated_nodes_.erase(u_it); 57 } else { 58 // This is a new node association, the sync node should be free. 59 // Remove node from free node pool and then associate it with the tab. 60 std::set<int>::iterator it = free_nodes_pool_.find(tab_node_id); 61 DCHECK(it != free_nodes_pool_.end()); 62 free_nodes_pool_.erase(it); 63 } 64 DCHECK(nodeid_tabid_map_.find(tab_node_id) == nodeid_tabid_map_.end()); 65 nodeid_tabid_map_[tab_node_id] = tab_id; 66 } 67 68 int TabNodePool::GetFreeTabNode() { 69 DCHECK_GT(machine_tag_.length(), 0U); 70 if (free_nodes_pool_.empty()) { 71 // Tab pool has no free nodes, allocate new one. 72 syncer::WriteTransaction trans(FROM_HERE, sync_service_->GetUserShare()); 73 syncer::ReadNode root(&trans); 74 if (root.InitByTagLookup(syncer::ModelTypeToRootTag(syncer::SESSIONS)) != 75 syncer::BaseNode::INIT_OK) { 76 LOG(ERROR) << kNoSessionsFolderError; 77 return kInvalidTabNodeID; 78 } 79 int tab_node_id = ++max_used_tab_node_id_; 80 std::string tab_node_tag = TabIdToTag(machine_tag_, tab_node_id); 81 syncer::WriteNode tab_node(&trans); 82 syncer::WriteNode::InitUniqueByCreationResult result = 83 tab_node.InitUniqueByCreation(syncer::SESSIONS, root, tab_node_tag); 84 if (result != syncer::WriteNode::INIT_SUCCESS) { 85 LOG(ERROR) << "Could not create new node with tag " 86 << tab_node_tag << "!"; 87 return kInvalidTabNodeID; 88 } 89 // We fill the new node with just enough data so that in case of a crash/bug 90 // we can identify the node as our own on re-association and reuse it. 91 tab_node.SetTitle(UTF8ToWide(tab_node_tag)); 92 sync_pb::SessionSpecifics specifics; 93 specifics.set_session_tag(machine_tag_); 94 specifics.set_tab_node_id(tab_node_id); 95 tab_node.SetSessionSpecifics(specifics); 96 97 // Grow the pool by 1 since we created a new node. 98 DVLOG(1) << "Adding sync node " << tab_node_id 99 << " to tab node id pool"; 100 free_nodes_pool_.insert(tab_node_id); 101 return tab_node_id; 102 } else { 103 // Return the next free node. 104 return *free_nodes_pool_.begin(); 105 } 106 } 107 108 void TabNodePool::FreeTabNode(int tab_node_id) { 109 TabNodeIDToTabIDMap::iterator it = nodeid_tabid_map_.find(tab_node_id); 110 DCHECK(it != nodeid_tabid_map_.end()); 111 nodeid_tabid_map_.erase(it); 112 FreeTabNodeInternal(tab_node_id); 113 } 114 115 void TabNodePool::FreeTabNodeInternal(int tab_node_id) { 116 DCHECK(free_nodes_pool_.find(tab_node_id) == free_nodes_pool_.end()); 117 free_nodes_pool_.insert(tab_node_id); 118 119 // If number of free nodes exceed kFreeNodesHighWatermark, 120 // delete sync nodes till number reaches kFreeNodesLowWatermark. 121 // Note: This logic is to mitigate temporary disassociation issues with old 122 // clients: http://crbug.com/259918. Newer versions do not need this. 123 if (free_nodes_pool_.size() > kFreeNodesHighWatermark) { 124 syncer::WriteTransaction trans(FROM_HERE, sync_service_->GetUserShare()); 125 for (std::set<int>::iterator free_it = free_nodes_pool_.begin(); 126 free_it != free_nodes_pool_.end();) { 127 syncer::WriteNode tab_node(&trans); 128 const std::string tab_node_tag = TabIdToTag(machine_tag_, *free_it); 129 if (tab_node.InitByClientTagLookup(syncer::SESSIONS, tab_node_tag) != 130 syncer::BaseNode::INIT_OK) { 131 LOG(ERROR) << "Could not find sync node with tag: " << tab_node_tag; 132 return; 133 } 134 free_nodes_pool_.erase(free_it++); 135 tab_node.Tombstone(); 136 if (free_nodes_pool_.size() <= kFreeNodesLowWatermark) { 137 return; 138 } 139 } 140 } 141 } 142 143 bool TabNodePool::IsUnassociatedTabNode(int tab_node_id) { 144 return unassociated_nodes_.find(tab_node_id) != unassociated_nodes_.end(); 145 } 146 147 void TabNodePool::ReassociateTabNode(int tab_node_id, 148 SessionID::id_type tab_id) { 149 // Remove from list of unassociated sync_nodes if present. 150 std::set<int>::iterator it = unassociated_nodes_.find(tab_node_id); 151 if (it != unassociated_nodes_.end()) { 152 unassociated_nodes_.erase(it); 153 } else { 154 // tab_node_id must be an already associated node. 155 DCHECK(nodeid_tabid_map_.find(tab_node_id) != nodeid_tabid_map_.end()); 156 } 157 nodeid_tabid_map_[tab_node_id] = tab_id; 158 } 159 160 SessionID::id_type TabNodePool::GetTabIdFromTabNodeId( 161 int tab_node_id) const { 162 TabNodeIDToTabIDMap::const_iterator it = nodeid_tabid_map_.find(tab_node_id); 163 if (it != nodeid_tabid_map_.end()) { 164 return it->second; 165 } 166 return kInvalidTabID; 167 } 168 169 void TabNodePool::DeleteUnassociatedTabNodes() { 170 syncer::WriteTransaction trans(FROM_HERE, sync_service_->GetUserShare()); 171 for (std::set<int>::iterator it = unassociated_nodes_.begin(); 172 it != unassociated_nodes_.end();) { 173 syncer::WriteNode tab_node(&trans); 174 const std::string tab_node_tag = TabIdToTag(machine_tag_, *it); 175 if (tab_node.InitByClientTagLookup(syncer::SESSIONS, tab_node_tag) != 176 syncer::BaseNode::INIT_OK) { 177 LOG(ERROR) << "Could not find sync node with tag: " << tab_node_tag; 178 } else { 179 tab_node.Tombstone(); 180 } 181 unassociated_nodes_.erase(it++); 182 } 183 DCHECK(unassociated_nodes_.empty()); 184 } 185 186 // Clear tab pool. 187 void TabNodePool::Clear() { 188 unassociated_nodes_.clear(); 189 free_nodes_pool_.clear(); 190 nodeid_tabid_map_.clear(); 191 max_used_tab_node_id_ = kInvalidTabNodeID; 192 } 193 194 size_t TabNodePool::Capacity() const { 195 return nodeid_tabid_map_.size() + unassociated_nodes_.size() + 196 free_nodes_pool_.size(); 197 } 198 199 bool TabNodePool::Empty() const { return free_nodes_pool_.empty(); } 200 201 bool TabNodePool::Full() { return nodeid_tabid_map_.empty(); } 202 203 void TabNodePool::SetMachineTag(const std::string& machine_tag) { 204 machine_tag_ = machine_tag; 205 } 206 207 } // namespace browser_sync 208