OpenSDN source code
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Pages
vm_stat_kvm.cc
Go to the documentation of this file.
1 /*
2  * Copyright (c) 2013 Juniper Networks, Inc. All rights reserved.
3  */
4 
5 #include <sys/times.h>
6 #include <sys/types.h>
7 #include <sys/wait.h>
8 #include <unistd.h>
9 #include <uve/vm_stat_kvm.h>
10 #include <uve/vm_stat_data.h>
11 #include <db/db.h>
12 #include <db/db_entry.h>
13 #include <db/db_table.h>
14 #include <base/address.h>
16 #include <cmn/agent.h>
17 #include <uve/vrouter_uve_entry.h>
18 #include <sstream>
19 #include <fstream>
20 #include <uve/agent_uve.h>
21 #include <uve/vm_uve_table.h>
22 
23 using namespace boost::uuids;
24 using namespace boost::asio;
25 
26 VmStatKvm::VmStatKvm(Agent *agent, const uuid &vm_uuid)
27  : VmStat(agent, vm_uuid) {
28 }
29 
31 }
32 
34  std::string tmp;
35  //Typical output from command
36  //Id: 1
37  //Name: instance-00000001
38  //UUID: 90cb7351-d2dc-4d8d-a216-2f460be183b6
39  //OS Type: hvm
40  //State: running
41  //CPU(s): 1
42  //CPU time: 13.4s
43  //Max memory: 2097152 KiB
44 
45  //Get 'CPU time' from the output
46  double cpu_stat = 0;
47  std::string cpu_stat_str, state_str, cpu_count_str;
48 
49  while (data_ >> tmp) {
50  if (tmp == "State:") {
51  data_ >> state_str;
52  }
53  if (tmp == "CPU(s):") {
54  data_ >> cpu_count_str;
55  }
56  if (tmp == "time:") {
57  data_ >> cpu_stat_str;
58  /* We expect 'State' and 'CPU(s)' fields to be present before
59  * 'CPU time' field. So break from the loop when we are done with
60  * reading of 'CPU time' field. */
61  break;
62  }
63  }
64 
65  vm_state_ = VrouterAgentVmState::VROUTER_AGENT_VM_UNKNOWN;
66  if (state_str.size()) {
67  if (state_str == "running") {
68  vm_state_ = VrouterAgentVmState::VROUTER_AGENT_VM_ACTIVE;
69  } else if (state_str == "paused") {
70  vm_state_ = VrouterAgentVmState::VROUTER_AGENT_VM_PAUSED;
71  } else if (state_str == "shut") {
72  vm_state_ = VrouterAgentVmState::VROUTER_AGENT_VM_SHUTDOWN;
73  }
74  }
76  if (cpu_count_str.size()) {
77  stringToInteger(cpu_count_str, vm_cpu_count_);
78  }
79  //Remove the last character from 'cpu_stat_str'
80  if (cpu_stat_str.size() >= 2) {
81  cpu_stat_str.erase(cpu_stat_str.size() - 1);
82  //Convert string to double
83  std::stringstream ss(cpu_stat_str);
84  ss >> cpu_stat;
85  }
86 
87  time_t now;
88  time(&now);
90  cpu_usage_ = (cpu_stat - prev_cpu_stat_)/
91  difftime(now, prev_cpu_snapshot_time_);
92  cpu_usage_ *= 100;
93  }
94 
95  prev_cpu_stat_ = cpu_stat;
97 
98  //Clear buffer
99  data_.str(" ");
100  data_.clear();
101 
102  //Trigger a request to start vcpu stat collection
103  GetVcpuStat();
104 }
105 
107  std::string tmp;
108  uint32_t index = 0;
109  std::vector<double> vcpu_usage;
110  //Read latest VCPU usage time
111  while(data_ >> tmp) {
112  if (tmp == "VCPU:") {
113  //Current VCPU index
114  data_ >> index;
115  }
116 
117  if (tmp == "time:") {
118  double usage = 0;
119  data_ >> usage;
120  vcpu_usage.push_back(usage);
121  }
122  }
123 
124  vcpu_usage_percent_.clear();
125  if (prev_vcpu_usage_.size() != vcpu_usage.size()) {
126  //In case a new VCPU get added
127  prev_vcpu_usage_ = vcpu_usage;
128  }
129 
130  time_t now;
131  time(&now);
132  //Calculate VCPU usage
134  for (uint32_t i = 0; i < vcpu_usage.size(); i++) {
135  double cpu_usage = (vcpu_usage[i] - prev_vcpu_usage_[i])/
136  difftime(now, prev_vcpu_snapshot_time_);
137  cpu_usage *= 100;
138  vcpu_usage_percent_.push_back(cpu_usage);
139  }
140  }
141 
142  prev_vcpu_usage_ = vcpu_usage;
144 
145  data_.str(" ");
146  data_.clear();
147  //Trigger a request to start mem stat
148  GetMemStat();
149 }
150 
152  if (pid_) {
153  std::ostringstream proc_file;
154  proc_file << "/proc/"<<pid_<<"/status";
155  std::ifstream file(proc_file.str().c_str());
156 
157  bool vmsize = false;
158  bool peak = false;
159  bool rss = false;
160  std::string line;
161  while (std::getline(file, line)) {
162  if (line.find("VmSize") != std::string::npos) {
163  std::stringstream vm(line);
164  std::string tmp; vm >> tmp; vm >> virt_memory_;
165  vmsize = true;
166  }
167  if (line.find("VmRSS") != std::string::npos) {
168  std::stringstream vm(line);
169  std::string tmp;
170  vm >> tmp;
171  vm >> mem_usage_;
172  rss = true;
173  }
174  if (line.find("VmPeak") != std::string::npos) {
175  std::stringstream vm(line);
176  std::string tmp; vm >> tmp; vm >> virt_memory_peak_;
177  peak = true;
178  }
179  if (rss && vmsize && peak)
180  break;
181  }
182  }
183 
184  data_.str(" ");
185  data_.clear();
186  GetDiskName();
187 }
188 
190  std::ostringstream cmd;
191  cmd << "virsh dominfo " << agent_->GetUuidStr(vm_uuid_);
192  ExecCmd(cmd.str(), boost::bind(&VmStatKvm::ReadCpuStat, this));
193 }
194 
196  std::ostringstream cmd;
197  cmd << "virsh vcpuinfo " << agent_->GetUuidStr(vm_uuid_);
198  ExecCmd(cmd.str(), boost::bind(&VmStatKvm::ReadVcpuStat, this));
199 }
200 
202  ReadMemStat();
203 }
204 
206  std::ostringstream cmd;
207  cmd << "virsh domblklist " << agent_->GetUuidStr(vm_uuid_) << " | grep "
209  ExecCmd(cmd.str(), boost::bind(&VmStatKvm::ReadDiskName, this));
210 }
211 
213  data_ >> disk_name_;
214  if (!disk_name_.empty()) {
215  GetDiskStat();
216  } else {
217  SendVmCpuStats();
218  StartTimer();
219  }
220 }
221 
223  std::ostringstream cmd;
224  cmd << "virsh domblkinfo " << agent_->GetUuidStr(vm_uuid_) << " "
225  << disk_name_;
226  ExecCmd(cmd.str(), boost::bind(&VmStatKvm::ReadDiskStat, this));
227 }
228 
230  bool disk_size_found = false, virtual_size_found = false;
231  std::string tmp;
232  std::string virtual_size_str, disk_size_str;
233  while (data_ >> tmp) {
234  if (tmp == "Capacity:") {
235  data_ >> virtual_size_str;
236  virtual_size_found = true;
237  } else if (tmp == "Allocation:") {
238  data_ >> disk_size_str;
239  disk_size_found = true;
240  }
241  if (virtual_size_found && disk_size_found) {
242  break;
243  }
244  }
245  if (virtual_size_str.size() >= 2) {
246  //Convert string to uint32_t
247  std::stringstream ss(virtual_size_str);
248  ss >> virtual_size_;
249  }
250 
251  if (disk_size_str.size() >= 2) {
252  //Convert string to uint32_t
253  std::stringstream ss(disk_size_str);
254  ss >> disk_size_;
255  }
256 
257  SendVmCpuStats();
258  StartTimer();
259 }
260 
262  std::string tmp;
263  while (data_ >> tmp) {
264  if (tmp == "actual") {
266  }
267  }
268  GetCpuStat();
269 }
270 
272  std::ostringstream cmd;
273  cmd << "virsh dommemstat " << agent_->GetUuidStr(vm_uuid_);
274  ExecCmd(cmd.str(), boost::bind(&VmStatKvm::ReadMemoryQuota, this));
275 }
276 
278  if (pid_ == 0) {
279  GetPid();
280  } else {
281  //Get CPU and memory stats
282  GetCpuStat();
283  }
284  return false;
285 }
286 
288  std::string tmp;
289  uint32_t pid;
290 
291  while (data_) {
292  data_ >> pid;
293  data_ >> tmp;
294  if (tmp.find("qemu") != std::string::npos ||
295  tmp.find("kvm") != std::string::npos) {
296  //Copy PID
297  pid_ = pid;
298  break;
299  }
300  //Flush out this line
301  data_.ignore(512, '\n');
302  }
303 
304  data_.str(" ");
305  data_.clear();
306  if (pid_) {
307  //Successfully read pid of process, collect other data
308  GetMemoryQuota();
309  } else {
310  retry_++;
311  //Retry after timeout
312  if (retry_ < kRetryCount) {
313  StartTimer();
314  }
315  }
316 }
317 
319  std::ostringstream cmd;
320  cmd << "ps -eo pid,cmd | grep " << agent_->GetUuidStr(vm_uuid_)
321  << " | grep instance-";
322  ExecCmd(cmd.str(), boost::bind(&VmStatKvm::ReadPid, this));
323 }
324 
326  GetPid();
327 }
std::vector< double > vcpu_usage_percent_
Definition: vm_stat.h:53
uint16_t vm_cpu_count_
Definition: vm_stat.h:68
uint32_t pid_
Definition: vm_stat.h:60
double cpu_usage_
Definition: vm_stat.h:50
time_t prev_cpu_snapshot_time_
Definition: vm_stat.h:51
void ExecCmd(std::string cmd, DoneCb cb)
Definition: vm_stat.cc:74
std::string disk_name_
Definition: vm_stat.h:65
void GetMemoryQuota()
Definition: vm_stat_kvm.cc:271
bool stringToInteger(const std::string &str, NumberType &num)
Definition: string_util.h:71
VrouterAgentVmState::type vm_state_
Definition: vm_stat.h:66
boost::uuids::uuid uuid
static const uint32_t kRetryCount
Definition: vm_stat.h:20
uint32_t disk_size_
Definition: vm_stat.h:64
Definition: vm_stat.h:17
void ReadPid()
Definition: vm_stat_kvm.cc:287
std::stringstream data_
Definition: vm_stat.h:56
void GetPid()
Definition: vm_stat_kvm.cc:318
void SendVmCpuStats()
Definition: vm_stat.cc:179
void Start()
Definition: vm_stat_kvm.cc:325
void GetDiskName()
Definition: vm_stat_kvm.cc:205
Definition: agent.h:358
uint32_t virt_memory_
Definition: vm_stat.h:46
void ReadDiskName()
Definition: vm_stat_kvm.cc:212
uint32_t mem_usage_
Definition: vm_stat.h:45
void ReadMemoryQuota()
Definition: vm_stat_kvm.cc:261
uint32_t virt_memory_peak_
Definition: vm_stat.h:47
uint32_t virtual_size_
Definition: vm_stat.h:63
std::vector< double > prev_vcpu_usage_
Definition: vm_stat.h:52
uint32_t vm_memory_quota_
Definition: vm_stat.h:48
void StartTimer()
Definition: vm_stat.cc:205
void ReadVcpuStat()
Definition: vm_stat_kvm.cc:106
void GetMemStat()
Definition: vm_stat_kvm.cc:201
double prev_cpu_stat_
Definition: vm_stat.h:49
void ReadCpuStat()
Definition: vm_stat_kvm.cc:33
void ReadMemStat()
Definition: vm_stat_kvm.cc:151
void ReadDiskStat()
Definition: vm_stat_kvm.cc:229
Agent * agent_
Definition: vm_stat.h:43
void GetVcpuStat()
Definition: vm_stat_kvm.cc:195
void GetCpuStat()
Definition: vm_stat_kvm.cc:189
static const uint16_t kInvalidCpuCount
Definition: vm_stat.h:21
void GetDiskStat()
Definition: vm_stat_kvm.cc:222
std::string GetUuidStr(boost::uuids::uuid uuid_val) const
Definition: agent.cc:98
uint32_t retry_
Definition: vm_stat.h:61
const boost::uuids::uuid vm_uuid_
Definition: vm_stat.h:44
bool TimerExpiry()
Definition: vm_stat_kvm.cc:277
time_t prev_vcpu_snapshot_time_
Definition: vm_stat.h:54
VmStatKvm(Agent *agent, const boost::uuids::uuid &vm_uuid)
Definition: vm_stat_kvm.cc:26